1 📕 READ ME

🚧 UNPUBLISHED AND ONGOING 🚧
The data in this script is currently unpublished, and the analysis is on going. There may be part of this script that are currently incomplete. This script has been made publicly available for easy of sharing with collaborators and colleagues who are interested in the progression of the project and for complete transparency.

SUMMARY
This R code is used to estimate the relationship between oxygen consumption (MO₂) and ambient oxygen partial pressure (PO₂) in the Common Galaxias (Galaxias maculatus). It also estimates the critical partial pressure of oxygen for aerobic metabolism (Pcrit), which is commonly understood as the threshold below which the oxygen consumption rate can no longer be sustained. The associated article is “The role of osmorespiratory compromise in hypoxia tolerance of the purportedly oxyconforming teleost Galaxias maculatus.” If you click the Code button in the top right of the HTML document, you can download the .Rmd file.

AIM
The article aims to test whether Galaxias maculatus can maintain oxygen consumption (MO₂) as ambient PO₂ falls and, if so, at what level it reaches the critical partial pressure of oxygen for aerobic metabolism (Pcrit).

AUTHORS
Timothy D. Clark [a]
Luis L. Kuchenmüller [a]
Elizabeth C. Hoots [a]
Maryane Gradito [a]
Jake M. Martin [a,b]
In no particular order

AFFILIATIONS
[a] School of Life and Environmental Sciences, Deakin University, Geelong, VIC, Australia
[b] School of Biological Sciences, Monash University, Clayton, Victoria, Australia

CONTRIBUTOR ROLES
🚧 To be added 🚧 Based on the Contributor Roles Taxonomy (CRediT)

DISCLAIMER
I (Jake Martin) am dyslexic. I have made an effort to review the script for grammatical errors, but some will likely remain. I apologise. Please reach out via the contact details below if anything is unclear.

2 📧 Contact

Jake M. Martin

📧 Email: jake.martin@deakin.edu.au

📧 Alt Email: jake.martin.research@gmail.com

🌐 Web: jakemartin.org

🐙 GitHub: JakeMartinResearch

3 ⚙️ Knit settings

Here we define our Knit settings, to make the output more user friendly, and to cache output for faster knitting.

#kniter seetting
knitr::opts_chunk$set(
message = FALSE,
warning = FALSE, # no warnings
cache = TRUE,# Cacheing to save time when kniting
tidy = TRUE
)

4 📦 Required packages

These are the R packages required for this script. You will need to install a package called pacman to run the p_load function.

# you need to install pacman first install.packages("pacman")
pacman::p_load("ggplot2", 
               "ggthemes", 
               "ggfortify", 
               "gtExtras", 
               "igraph",
               "dagitty",
               "ggdag",
               "ggridges",
               "gghalves",
               "ggExtra",
               "gridExtra",
               "corrplot",
               "RColorBrewer", 
               "gt", 
               "gtsummary",
               "grid",
               "plotly",
               "bayesplot", # data visualisation
               
                "tidyverse", 
               "janitor", 
               "readxl", 
               "broom.mixed", 
               "data.table", 
               "devtools",
               "hms", # data tidy
               
               "marginaleffects", 
               "brms", 
               "rstan", 
               "performance", 
               "emmeans", 
               "tidybayes", 
               "vegan",
               "betareg",
               "lme4", 
               "car", 
               "lmerTest",
               "qqplotr",
               "respirometry",
               "mclust",
               "furrr", # modelling 
              
               
               "datawizard", 
               "SRS", # data manipulation 
               
               "sessioninfo",
               "pander" # reproducibility
                       )

5 🔧 Custom functions

Here are some custom function used within this script.

bayes_incremental_regression_by_id(): A custom function to build Bayesian incremental regressions. It is formatted for running a list of sub group models (ids) in parallel with 4 cores. It uses brm() with a Gaussian error function.

Use: The function takes an grouping factor/id that will be used to filter the data for the regression, if none is proved it will use all data (id_i), the column name of the grouping factor/charter in the data frame, if none is proved it will use all data (id_name), the data frame (data), the predictor of interest (predictor), the response of interest (response), the random seed number for model reproducibility (seed_number), where you would like to save the model outputs (save_models = logical argument), the output directory for the saved rds models (mod_output_wd)

bayes_incremental_regression_by_id(id_i = “your_id”, id_name = “column name for ids”, data = “your data”, predictor = “predictor variable”, response = “response variable”, seed_number = integer, save_models = TRUE/FALSE, mod_output_wd = “directory for model output”)

# Instead we could also use a distributional regression approach, by
# specifically modelling the variance by DO (e.g. sigma ~ DO). Weighting may
# not be required in this case, I don't think higher density of vaules in a
# given space will effect Bayesian estimates like it does in frequentist
# models. See discourse https://discourse.mc-stan.org/t/weights-in-brm/4278

bayes_incremental_regression_by_id <- function(id_i, id_name, data, predictor, response,
    seed_number, save_models, mod_output_wd) {
    # Initiate an empty list to store models
    models <- list()

    # Check if id_name is missing, NULL, or blank, and assign NA if so
    if (missing(id_name) || is.null(id_name) || id_name == "") {
        id_name <- NA
    }

    # Check if id_i is missing, NULL, or blank, and assign NA if so
    if (missing(id_i) || is.null(id_i) || id_i == "") {
        id_name <- NA
    }

    # Filter data for the current ID if id_name is given as a factor or
    # character and id_i is defined
    df_i <- data %>%
        dplyr::filter(if (!is.na(id_i) && (is.factor(data[[id_name]]) || is.character(data[[id_name]]))) {
            !!rlang::sym(id_name) == id_i
        } else {
            TRUE
        })

    # Dynamically create formulas
    formula_lm_0 <- reformulate("1", response)
    formula_lm_1 <- reformulate(predictor, response)
    formula_lm_2 <- reformulate(sprintf("poly(%s, 2)", predictor), response)
    formula_lm_3 <- reformulate(sprintf("poly(%s, 3)", predictor), response)

    # Fit and store models in the list
    models[[paste0(id_i, "_lm_0")]] <- brm(bf(formula_lm_0, family = gaussian()),
        data = df_i, cores = 4, seed = seed_number, save_pars = save_pars(all = save_models),
        sample_prior = FALSE, silent = TRUE, file = paste0(mod_output_wd, "/", id_i,
            "_lm_0"))

    models[[paste0(id_i, "_lm_1")]] <- brm(bf(formula_lm_1, family = gaussian()),
        data = df_i, cores = 4, seed = seed_number, save_pars = save_pars(all = save_models),
        sample_prior = FALSE, silent = TRUE, file = paste0(mod_output_wd, "/", id_i,
            "_lm_1"))

    models[[paste0(id_i, "_lm_2")]] <- brm(bf(formula_lm_2, family = gaussian()),
        data = df_i, cores = 4, seed = seed_number, save_pars = save_pars(all = save_models),
        sample_prior = FALSE, silent = TRUE, file = paste0(mod_output_wd, "/", id_i,
            "_lm_2"))

    models[[paste0(id_i, "_lm_3")]] <- brm(bf(formula_lm_3, family = gaussian()),
        data = df_i, cores = 4, seed = 143019, save_pars = save_pars(all = save_models),
        sample_prior = FALSE, silent = TRUE, file = paste0(mod_output_wd, "/", id_i,
            "_lm_3"))

    # Return the list of models for the current ID
    return(models)
}

load_rds9(): A custom function to load all rds models in a directory and store in a list. Takes a directory with .rds files

load_rds <- function(model_dw) {
    # List all .rds files in the directory
    model_file_list <- list.files(path = model_dw, pattern = "\\.rds$", full.names = TRUE)

    # Initialise an empty list to store models
    model_store_list <- list()

    # Iterate through each file and load the RDS
    for (mod_i in model_file_list) {
        mod <- readRDS(file = mod_i)  # Read the RDS file
        model_name <- tools::file_path_sans_ext(basename(mod_i))  # Extract the file name without extension
        model_store_list[[model_name]] <- mod  # Store it in the list
    }

    # Return the list of models
    return(model_store_list)
}

incremental_regression_bayes_fits(): A custom function for pulling model fits, loo and r2 using loo() and bayes_R2(), respectively. Takes a list of brm models.

# Define Function to Process the data for each ID
incremental_regression_bayes_fits <- function(models) {
  
  loo_results_list <- list()
  
  # Iterate over the names of the models
  for (mod_name in names(models)) {
    # Extract the model
    mod_i <- models[[mod_name]]
    
    # Compute LOO results
    mod_loo_results_i <- loo::loo(mod_i)
    
    # Extract relevant LOO metrics
    elpd_loo_i <- mod_loo_results_i$elpd_loo
    p_loo_i <- mod_loo_results_i$p_loo
    looic_i <- mod_loo_results_i$looic
    
    # Create a data frame with metrics
    df_i <- data.frame(
      elpd_loo = elpd_loo_i,
      p_loo = p_loo_i,
      looic = looic_i,
      model = mod_name
    )
    
    est_i <- tidy(mod_i, effects = "fixed", conf.int = TRUE) %>% 
      dplyr::select(term, estimate, conf.low, conf.high) %>% 
      tidyr::pivot_wider(
        names_from = term,                # Use `term` as column names
        values_from = c(estimate, conf.low, conf.high),  # Values to pivot
        names_sep = "_"                   # Add a separator to column names
      )
    
    df_i <- cbind(df_i, est_i)
    
    # Store the data frame in the list
    loo_results_list[[mod_name]] <- df_i
  }
  
  # Combind 
  loo_results_combined <- bind_rows(loo_results_list)
  
  # Get R2 
  r2_results <- map_dfr(models, ~ as.data.frame(bayes_R2(.x)), .id = "model") %>%
    tibble::remove_rownames()
  
  # Combind R2 and loo results 
  model_fit_df <- dplyr::full_join(loo_results_combined, r2_results, by = "model") %>% 
    dplyr::select(model, everything()) %>% 
    dplyr::rename(r2 = Estimate,
                  r2_error = Est.Error,
                  r2_q2.5 = Q2.5,
                  r2_q97.5 = Q97.5) %>% 
    dplyr::mutate(id = sub("_(lm_\\d+)$", "", model),
                  model_type = sub("^.*_(lm_\\d+)$", "\\1", model))
  
  return(model_fit_df)
}

bayes_mod_predictions(): This function extracts the predicted values using fitted() from a list of models and combines them with the original data file used for the model. These are the posterior mean fitted values (i.e. the expected value of the response variable given the predictor variables and the estimated posterior distributions of the parameters) for each observation in the dataset, along with 95% credible intervals.

bayes_mod_predictions <- function(models, original_data) {

    prediction_list <- list()

    for (mod_name in names(models)) {
        # Extract mod
        mod_i <- models[[mod_name]]

        # Make mode predictions
        model_predictions_i <- as.data.frame(fitted(mod_i, summary = TRUE)) %>%
            dplyr::mutate(model = mod_name, id = sub("_(lm_\\d+)$", "", mod_name),
                model_type = sub("^.*_(lm_\\d+)$", "\\1", mod_name)) %>%
            dplyr::rename(pred_lower = Q2.5, pred_upper = Q97.5, predicted = Estimate,
                pred_error = Est.Error) %>%
            dplyr::select(model, everything())

        id_i <- model_predictions_i$id[1]

        original_data_i <- original_data %>%
            dplyr::filter(id == id_i) %>%
            dplyr::select(-id)

        model_predictions_original_i <- cbind(model_predictions_i, original_data_i)

        prediction_list[[mod_name]] <- model_predictions_original_i
    }
    predictions_df <- bind_rows(prediction_list)
    return(predictions_df)
}

calcSMR(): authored by Denis Chabot used to estimate SMR with several different methods Claireaux and Chabot (2016) [1]

[1] Claireaux, G. and Chabot, D. (2016) Responses by fishes to environmental hypoxia: integration through Fry’s concept of aerobic metabolic scope. Journal of Fish Biology https://doi.org/10.1111/jfb.12833

calcSMR = function(Y, q = c(0.1, 0.15, 0.2, 0.25, 0.3), G = 1:4) {
    u = sort(Y)
    the.Mclust <- Mclust(Y, G = G)
    cl <- the.Mclust$classification
    # sometimes, the class containing SMR is not called 1 the following
    # presumes that when class 1 contains > 10% of cases, it contains SMR,
    # otherwise we take class 2
    cl2 <- as.data.frame(table(cl))
    cl2$cl <- as.numeric(levels(cl2$cl))
    valid <- cl2$Freq >= 0.1 * length(time)
    the.cl <- min(cl2$cl[valid])
    left.distr <- Y[the.Mclust$classification == the.cl]
    mlnd = the.Mclust$parameters$mean[the.cl]
    CVmlnd = sd(left.distr)/mlnd * 100
    quant = quantile(Y, q)
    low10 = mean(u[1:10])
    low10pc = mean(u[6:(5 + round(0.1 * (length(u) - 5)))])
    # remove 5 outliers, keep lowest 10% of the rest, average Herrmann & Enders
    # 2000
    return(list(mlnd = mlnd, quant = quant, low10 = low10, low10pc = low10pc, cl = cl,
        CVmlnd = CVmlnd))
}

calcO2crit(): authored by Denis Chabot used to estimate O2crit (Pcript). Claireaux and Chabot (2016) [1]

Note: O2 is assumed to be in percentage of dissolved oxygen (DO)

calcO2crit <- function(Data, SMR, lowestMO2 = NA, gapLimit = 4, max.nb.MO2.for.reg = 20) {
    # AUTHOR: Denis Chabot, Institut Maurice-Lamontagne, DFO, Canada first
    # version written in June 2009 last updated in January 2015
    method = "LS_reg"  # will become 'through_origin' if intercept is > 0
    if (is.na(lowestMO2))
        lowestMO2 = quantile(Data$MO2[Data$DO >= 80], p = 0.05)
    # Step 1: identify points where MO2 is proportional to DO
    geqSMR = Data$MO2 >= lowestMO2
    pivotDO = min(Data$DO[geqSMR])
    lethal = Data$DO < pivotDO
    N_under_SMR = sum(lethal)  # points available for regression?
    final_N_under_SMR = lethal  # some points may be removed at Step 4
    lastMO2reg = Data$MO2[Data$DO == pivotDO]  # last MO2 when regulating
    if (N_under_SMR > 1)
        theMod = lm(MO2 ~ DO, data = Data[lethal, ])
    # Step 2, add one or more point at or above SMR 2A, when there are fewer
    # than 3 valid points to calculate a regression
    if (N_under_SMR < 3) {
        missing = 3 - sum(lethal)
        not.lethal = Data$DO[geqSMR]
        DOlimit = max(sort(not.lethal)[1:missing])  # highest DO acceptable
        # to reach a N of 3
        addedPoints = Data$DO <= DOlimit
        lethal = lethal | addedPoints
        theMod = lm(MO2 ~ DO, data = Data[lethal, ])
    }
    # 2B, add pivotDO to the fit when Step 1 yielded 3 or more values?
    if (N_under_SMR >= 3) {
        lethalB = Data$DO <= pivotDO  # has one more value than 'lethal'
        regA = theMod
        regB = lm(MO2 ~ DO, data = Data[lethalB, ])
        large_slope_drop = (coef(regA)[2]/coef(regB)[2]) > 1.1  # arbitrary
        large_DO_gap = (max(Data$DO[lethalB]) - max(Data$DO[lethal])) > gapLimit
        tooSmallMO2 = lastMO2reg < SMR
        if (!large_slope_drop & !large_DO_gap & !tooSmallMO2)
            {
                lethal = lethalB
                theMod = regB
            }  # otherwise we do not accept the additional point
    }
    # Step 3 if the user wants to limit the number of points in the regression
    if (!is.na(max.nb.MO2.for.reg) & sum(lethal) > max.nb.MO2.for.reg) {
        Ranks = rank(Data$DO)
        lethal = Ranks <= max.nb.MO2.for.reg
        theMod = lm(MO2 ~ DO, data = Data[lethal, ])
        final_N_under_SMR = max.nb.MO2.for.reg
    }
    # Step 4
    predMO2 = as.numeric(predict(theMod, data.frame(DO = Data$DO)))
    Data$delta = (Data$MO2 - predMO2)/predMO2 * 100  # residuals set to zero
    # when below pivotDO
    Data$delta[Data$DO < pivotDO | lethal] = 0
    tol = 0  # any positive residual is unacceptable
    HighValues = Data$delta > tol
    Ranks = rank(-1 * Data$delta)
    HighMO2 = HighValues & Ranks == min(Ranks)  # keep largest residual
    if (sum(HighValues) > 0)
        {
            nblethal = sum(lethal)
            Data$W = NA
            Data$W[lethal] = 1/nblethal
            Data$W[HighMO2] = 1
            theMod = lm(MO2 ~ DO, weight = W, data = Data[lethal | HighMO2, ])
            # This new regression is always an improvement, but there can still
            # be points above the line, so we repeat
            predMO2_2 = as.numeric(predict(theMod, data.frame(DO = Data$DO)))
            Data$delta2 = (Data$MO2 - predMO2_2)/predMO2_2 * 100
            Data$delta2[Data$DO < pivotDO] = 0
            tol = Data$delta2[HighMO2]
            HighValues2 = Data$delta2 > tol
            if (sum(HighValues2) > 0)
                {
                  Ranks2 = rank(-1 * Data$delta2)
                  HighMO2_2 = HighValues2 & Ranks2 == 1  # keep the largest residual
                  nblethal = sum(lethal)
                  Data$W = NA
                  Data$W[lethal] = 1/nblethal
                  Data$W[HighMO2_2] = 1
                  theMod2 = lm(MO2 ~ DO, weight = W, data = Data[lethal | HighMO2_2,
                    ])
                  # is new slope steeper than the old one?
                  if (theMod2$coef[2] > theMod$coef[2]) {
                    theMod = theMod2
                    HighMO2 = HighMO2_2
                  }
                }  # end second search for high value
        }  # end first search for high value
    Coef = coefficients(theMod)
    # Step 5, check for positive intercept
    AboveOrigin = Coef[1] > 0
    # if it is, we use a regression that goes through the origin
    if (AboveOrigin) {
        theMod = lm(MO2 ~ DO - 1, data = Data[lethal, ])
        Coef = c(0, coefficients(theMod))  # need to add the intercept (0)
        # manually to have a pair of coefficients
        method = "through_origin"
        HighMO2 = rep(FALSE, nrow(Data))  # did not use the additional value
        # from Step 4
    }
    po2crit = as.numeric(round((SMR - Coef[1])/Coef[2], 1))
    sum_mod = summary(theMod)
    anov_mod = anova(theMod)
    O2CRIT = list(o2crit = po2crit, SMR = SMR, Nb_MO2_conforming = N_under_SMR, Nb_MO2_conf_used = final_N_under_SMR,
        High_MO2_required = sum(HighMO2) == 1, origData = Data, Method = method,
        mod = theMod, r2 = sum_mod$r.squared, P = anov_mod$"Pr(>F)", lethalPoints = which(lethal),
        AddedPoints = which(HighMO2))
}  # end function

plotO2crit(): authored by Denis Chabot, used to plot the modes used for the calcO2crit() function. Claireaux and Chabot (2016) [1]

Note: I added abline(h=lowestMO2, col=“pink”) so that I could visualise the lowestMO2 position

plotO2crit <- function(o2critobj, plotID = "", Xlab = "Dissolved oxygen (% sat.)",
    Ylab = "dotitalumol", smr.cex = 0.9, o2crit.cex = 0.9, plotID.cex = 1.2, Transparency = T,
    ...) {
    # AUTHOR: Denis Chabot, Institut Maurice-Lamontagne, DFO, Canada first
    # version written in June 2009 last updated 2015-02-09 for R plotting
    # devices that do not support transparency (e.g., postscript), set
    # Transparency to FALSE
    smr = o2critobj$SMR
    if (Ylab %in% c("dotitalumol", "italumol", "dotumol", "umol", "dotitalmg", "italmg",
        "dotmg", "mg")) {
        switch(Ylab, dotitalumol = {
            mo2.lab = expression(paste(italic(dot(M))[O[2]], " (", mu, "mol ", O[2],
                " ", min^-1, " ", kg^-1, ")"))
        }, italumol = {
            mo2.lab = expression(paste(italic(M)[O[2]], " (", mu, "mol ", O[2], " ",
                min^-1, " ", kg^-1, ")"))
        }, dotumol = {
            mo2.lab = expression(paste(dot(M)[O[2]], " (", mu, "mol ", O[2], " ",
                min^-1, " ", kg^-1, ")"))
        }, umol = {
            mo2.lab = expression(paste(M[O[2]], " (", mu, "mol ", O[2], " ", min^-1,
                " ", kg^-1, ")"))
        }, dotitalmg = {
            mo2.lab = expression(paste(italic(dot(M))[O[2]], " (mg ", O[2], " ",
                h^-1, " ", kg^-1, ")"))
        }, italmg = {
            mo2.lab = expression(paste(italic(M)[O[2]], " (mg ", O[2], " ", h^-1,
                " ", kg^-1, ")"))
        }, dotmg = {
            mo2.lab = expression(paste(dot(M)[O[2]], " (mg ", O[2], " ", h^-1, " ",
                kg^-1, ")"))
        }, mg = {
            mo2.lab = expression(paste(M[O[2]], " (mg ", O[2], " ", h^-1, " ", kg^-1,
                ")"))
        })
    } else mo2.lab = Ylab
    if (Transparency) {
        Col = c(rgb(0, 0, 0, 0.7), "red", "orange")
    } else {
        Col = c(grey(0.3), "red", "orange")
    }
    Data = o2critobj$origData
    lowestMO2 = quantile(Data$MO2[Data$DO >= 80], p = 0.05)  # I added this
    Data$Color = Col[1]
    Data$Color[o2critobj$lethalPoints] = Col[2]
    Data$Color[o2critobj$AddedPoints] = Col[3]
    # ordinary LS regression without added points: blue line, red symbols
    # ordinary LS regression with added points: blue line, red & orange symbols
    # regression through origin: green dotted line, red symbols
    line.color = ifelse(o2critobj$Method == "LS_reg", "blue", "darkgreen")
    line.type = ifelse(o2critobj$Method == "LS_reg", 1, 3)
    limX = c(0, max(Data$DO))
    limY = c(0, max(Data$MO2))
    plot(MO2 ~ DO, data = Data, xlim = limX, ylim = limY, col = Data$Color, xlab = Xlab,
        ylab = mo2.lab, ...)
    coord <- par("usr")
    if (plotID != "") {
        text(0, coord[4], plotID, cex = plotID.cex, adj = c(0, 1.2))
    }
    abline(h = lowestMO2, col = "pink")  # I added this
    abline(h = smr, col = "orange")
    text(coord[1], smr, "SMR", adj = c(-0.1, 1.3), cex = smr.cex)
    text(coord[1], smr, round(smr, 1), adj = c(-0.1, -0.3), cex = smr.cex)
    if (!is.na(o2critobj$o2crit)) {
        abline(o2critobj$mod, col = line.color, lty = line.type)
        segments(o2critobj$o2crit, smr, o2critobj$o2crit, coord[3], col = line.color,
            lwd = 1)
        text(x = o2critobj$o2crit, y = 0, o2critobj$o2crit, col = line.color, cex = o2crit.cex,
            adj = c(-0.1, 0.5))
    }
}  # end of function

6 📂 Directories

6.1 Input

📥 input_data_wd: Directory for the metadata

wd <- getwd()
input_data_wd <- paste0(wd, "./input-data")  # creates a variable with the name of the wd we want to use

📥 mod_data_wd: Directory for model output data estimated slopes

mod_data_wd <- paste0(wd, "./mod-data")

6.2 Output

📤 output_fig_wd: this is where we will put the figures

output_fig_wd <- paste0(wd, "./output-fig")
if (!dir.exists("output-fig")) {
    dir.create("output-fig")
}

📤 output_mods_wd: this is where we will put the models

output_mods_wd <- paste0(wd, "./output-mod")
if (!dir.exists("output-mod")) {
    dir.create("output-mod")
}

7 💿 Data

7.1 Slopes (MO₂)

💿 slope_df: We have imported the slopes extracted in LabChart during each phase of the experiment

 setwd(input_data_wd)
# 
# # Get the names of all sheets in the Excel file
sheet_names <- excel_sheets("labchart-all-dates-2025.xlsx")
all_trials_select <- c("start_date", "order", "phase", "cycle", "date", "time")
slope_list <- list()

for (sheet in sheet_names) {

  df <- read_excel("labchart-all-dates-2025.xlsx", sheet = sheet) %>% 
  dplyr::rename_with(tolower)
  
a_name <- paste0("a_", tolower(sheet))
a_df <- df %>%
  dplyr::select(starts_with('a'), all_trials_select) %>% 
  dplyr::rename(temp = a_temp) %>% 
  dplyr::mutate(across(starts_with('a'), as.numeric)) %>% 
  pivot_longer(
    cols = starts_with('a'), # Select all columns to pivot
    names_to = c("chamber_id", ".value"), # Separate column names into 'id' and other variables
    names_sep = "_"
  ) %>%
  dplyr::mutate(respirometer_group = "a") # Add a new column with a fixed value

slope_list[[a_name]]<- a_df

b_name <- paste0("b_", tolower(sheet))
b_df <- df %>% 
  dplyr::select(starts_with('b'), all_trials_select) %>% 
  dplyr::rename(temp = b_temp) %>% 
  dplyr::mutate(across(starts_with('b'), as.numeric)) %>% 
  pivot_longer(
    cols = starts_with('b'), # Select all columns to pivot
    names_to = c("chamber_id", ".value"), # Separate column names into 'id' and other variables
    names_sep = "_"
  ) %>% 
    dplyr::mutate(respirometer_group = "b")

slope_list[[b_name]] <- b_df

c_name <- paste0("c_", tolower(sheet))
c_df <- df %>% 
  dplyr::select(starts_with('c'), all_trials_select) %>% 
  dplyr::rename(temp = c_temp,
                i_cycle = cycle) %>% 
  dplyr::mutate(across(starts_with('c'), as.numeric)) %>%
  pivot_longer(
    cols = starts_with('c'), # Select all columns to pivot
    names_to = c("chamber_id", ".value"), # Separate column names into 'id' and other variables
    names_sep = "_"
  ) %>% 
    dplyr::mutate(respirometer_group = "c") %>% 
  dplyr::rename(cycle = i_cycle)

slope_list[[c_name]] <- c_df

d_name <- paste0("d_", tolower(sheet))
d_df <- df %>% 
  dplyr::select(starts_with('d'), all_trials_select) %>% 
  dplyr::rename(temp = d_temp,
                i_date = date) %>% 
  dplyr::mutate(across(starts_with('d'), as.numeric)) %>%
  pivot_longer(
    cols = starts_with('d'), # Select all columns to pivot
    names_to = c("chamber_id", ".value"), # Separate column names into 'id' and other variables
    names_sep = "_"
  ) %>% 
    dplyr::mutate(respirometer_group = "d") %>% 
  dplyr::rename(date = i_date)

slope_list[[d_name]] <- d_df
}


slope_df <- bind_rows(slope_list) %>% 
  dplyr::mutate(resp_cat_date = paste0(respirometer_group, "_", start_date),
                chamber_n = str_extract(chamber_id, "\\d+"),
                id_prox = paste0(resp_cat_date, "_", chamber_n),
                time_hms = as_hms(time*3600),
                date_chr = format(date, "%d/%m/%Y")
                )

7.2 Fish metadata

💿 metadata: This is the meta data for each chamber

Note: We are also adding volume based on chamber type.

setwd(input_data_wd)

metadata <- read_excel("fish-meta.xlsx", na = "NA") %>%
    dplyr::mutate(id_split = id) %>%
    tidyr::separate(id_split, into = c("respirometer_group", "salinity_group", "start_date",
        "chamber"), sep = "_") %>%
    dplyr::mutate(volume = dplyr::case_when(chamber_type == "L" ~ 0.3, chamber_type ==
        "M_M" ~ 0.105, chamber_type == "M_NM" ~ 0.11, chamber_type == "S" ~ 0.058,
        chamber_type == "SM" ~ 0.075, chamber_type == "D3" ~ 0.055, TRUE ~ NA), id_prox = paste0(respirometer_group,
        "_", start_date, "_", chamber), rel_size = mass/volume)

7.3 Urbina, Glover, and Forster (2012)

💿 urbina_et_al_2012: This is the mean level data extracted from Urbina, Glover, and Forster (2012)[2] Figure 1a. We used the metaDigitise package in R to extract the data [3].

setwd(input_data_wd)

urbina_et_al_2012 <- read.csv("urbina-et-al-2012-fig1a.csv")

7.4 Pcrit viusal inspection

💿 pcrit_check: This data frame is a list of all fish that had higher order polynomials (2nd and 3rd order; see ‘Incremental regression analyses’ for details) and subsequently had Pcrit modelled (see ‘Pcrit model’ for details). In this data we have checked each Pcrit model and confirmed visual if a Pcrit is present.

setwd(input_data_wd)

pcrit_check <- read_excel("pcrit-check.xlsx", na = "NA")

7.4.1 Combinding metadata

Adding the meta data to the slopes data frame

slope_df_2 <- slope_df %>%
    dplyr::select(-start_date, -respirometer_group) %>%
    left_join(metadata, by = "id_prox") %>%
    dplyr::mutate(light_dark = if_else(time_hms >= as.hms("07:00:00") & time_hms <
        as.hms("19:00:00"), "light", "dark")) %>%
    dplyr::arrange(id)

8 🧼 Data tidy

8.1 Numbers

We have 64 fish with MO₂ data

n <- slope_df_2 %>%
    dplyr::filter(chamber_condition == "fish") %>%
    dplyr::distinct(id) %>%
    nrow(.)

paste0("n = ", n)
## [1] "n = 64"

With 48 from the 0 ppt and 48 from 9 ppt groups

slope_df_2 %>%
    dplyr::filter(chamber_condition == "fish") %>%
    dplyr::group_by(salinity_group) %>%
    dplyr::reframe(`n total` = length(unique(id))) %>%
    gt() %>%
    cols_label(salinity_group = "Salinity group") %>%
    cols_align(align = "center", columns = everything())
Salinity group n total
0 33
9 31

8.2 Fish size

Here we calculate the mean length and size of fish used in the experiment.

mass_length <- slope_df_2 %>%
    dplyr::filter(chamber_condition == "fish") %>%
    dplyr::group_by(id) %>%
    dplyr::sample_n(1) %>%
    dplyr::ungroup() %>%
    dplyr::reframe(x_mass = round(mean(mass, na.rm = TRUE), 3), min_mass = round(min(mass,
        na.rm = TRUE), 3), max_mass = round(max(mass, na.rm = TRUE), 3), x_length = round(mean(length,
        na.rm = TRUE), 2), min_length = round(min(length, na.rm = TRUE), 2), max_length = round(max(length,
        na.rm = TRUE), 2))

mass_mean <- mass_length %>%
    pull(x_mass)

mass_min <- mass_length %>%
    pull(min_mass)

mass_max <- mass_length %>%
    pull(max_mass)

length_mean <- mass_length %>%
    pull(x_length)

length_min <- mass_length %>%
    pull(min_length)

length_max <- mass_length %>%
    pull(max_length)

paste0("The mean mass of fish was ", mass_mean, " g (range: ", mass_min, "-", mass_max,
    ")", ", and the mean length was ", length_mean, " mm (range: ", length_min, "-",
    length_max, ")")
## [1] "The mean mass of fish was 0.532 g (range: 0.21-1.6), and the mean length was 50.41 mm (range: 40-70)"

8.3 Filtering trials

We will remove 6 trials which had errors. These are as follows:

  • a_0_25nov_3 fish died during trial
  • b_0_26nov_4 flat lined early
  • c_0_22nov_2 chamber was opened early
  • c_9_26nov_2 chamber was opened early
  • c_9_26nov_4 chamber was opened early
  • d_9_27nov_3 sensor was jumpy and end points were hard to confidently ID visually
remove_trial_error <- c("a_0_25nov_3", "b_0_26nov_4", "c_0_22nov_2", "c_9_26nov_2",
    "c_9_26nov_4", "d_9_27nov_3")

slope_df_filter <- slope_df_2 %>%
    dplyr::filter(!(id %in% remove_trial_error))

We now have 58 fish with MO2 data

n <- slope_df_filter %>%
    dplyr::filter(chamber_condition == "fish") %>%
    dplyr::distinct(id) %>%
    nrow(.)

paste0("n = ", n)
## [1] "n = 58"

With 30 in the 0 ppt group and 28 in the 9 ppt group

slope_df_filter %>%
    dplyr::filter(chamber_condition == "fish") %>%
    dplyr::group_by(salinity_group) %>%
    dplyr::reframe(`n total` = length(unique(id))) %>%
    gt() %>%
    cols_label(salinity_group = "Salinity group") %>%
    cols_align(align = "center", columns = everything())
Salinity group n total
0 30
9 28

8.4 Filtering MO₂ estimates

Here we apply the following filters to the MO₂ data:

  • Remove the first 5 SMR cycles (burn in)
  • Remove all positive raw slopes
  • Remove all MO₂ calculated using less then 60 data points (5 min)
  • Remove all MO₂ calculated if O₂ increases in a closed phase (i.e. trial has ended)

Check positive values for MO₂ before removing.

slope_tidy_remove_flush <- slope_df_filter %>%
  dplyr::filter(phase != "smr", chamber_condition == "fish") %>% 
  dplyr::group_by(id) %>%
  dplyr::arrange(id, order) %>%  # Ensure the data is sorted within each group
  dplyr::mutate(o2_diff = if_else(row_number() == 1, 0, o2 - lag(o2)), # Calculate the difference in 'o2'
                o2_diff_cumsum = cumsum(o2_diff > 1)) %>%  # Checks first occurrence and sums 
  dplyr::filter(o2_diff_cumsum == 0) %>%  # Keep rows until the first jump > 1
  dplyr::ungroup() %>%
  dplyr::select(-o2_diff, -o2_diff_cumsum)

postive_slopes <- slope_tidy_remove_flush %>%
  dplyr::filter(chamber_condition == "fish" & mo2corr > 0)

list_postive_all <- slope_df_filter %>% 
  dplyr::filter(chamber_condition == "fish" & mo2corr > 0 & phase == "smr") %>% 
  dplyr::bind_rows(., postive_slopes) %>% 
  dplyr::distinct(id) %>% 
  dplyr::pull(id)

print(paste0("There are ", length(list_postive_all), " fish with postive slopes. These fish are: ", paste(list_postive_all, collapse = ", "), "."))
## [1] "There are 22 fish with postive slopes. These fish are: a_9_22nov_1, d_0_21nov_3, a_0_24nov_1, a_0_25nov_1, b_0_24nov_1, b_0_24nov_3, b_0_25nov_1, b_0_25nov_4, b_9_21nov_1, b_9_21nov_2, b_9_21nov_3, b_9_22nov_1, b_9_22nov_2, b_9_22nov_3, c_0_21nov_1, c_0_21nov_2, c_0_22nov_4, c_9_24nov_3, c_9_25nov_3, d_0_22nov_2, d_9_25nov_2, d_9_25nov_3."

Filtering the MO₂ data

cycle_burn <- 0:4

slope_df_filter_1 <- slope_df_filter %>%
  dplyr::filter(!(cycle %in% cycle_burn) & 
                  mo2corr < 0 & 
                  n > 60 &
                  chamber_condition == "fish"
                )
  
# Filter out the end flush
slope_tidy_closed <- slope_df_filter_1 %>%
  dplyr::filter(phase != "smr") %>% 
  dplyr::group_by(id) %>%
  dplyr::arrange(id, order) %>%  # Ensure the data is sorted within each group
  dplyr::mutate(o2_diff = if_else(row_number() == 1, 0, o2 - lag(o2)), # Calculate the difference in 'o2'
                o2_diff_cumsum = cumsum(o2_diff > 1)) %>%  # Checks first occurrence and sums 
  dplyr::filter(o2_diff_cumsum == 0) %>%  # Keep rows until the first jump > 1
  dplyr::ungroup() %>%
  dplyr::select(-o2_diff, -o2_diff_cumsum)

slope_tidy_smr <- slope_df_filter_1 %>% 
  dplyr::filter(phase == "smr")

slope_df_filter_2 <- rbind(slope_tidy_smr, slope_tidy_closed) %>% 
  dplyr::arrange(id, order)

8.5 Calculating SMR

We will calculate SMR using calcSMR function by Chabot, Steffensen and Farrell (2016)[1]. Specifically, we use mean of the lowest normal distribution (MLND) where CVmlnd < 5.4 and the mean of the lower 20% quantile (q0.2) were CVmlnd > 5.4. If CVmlnd is not calculated we have used q0.2.

labchart_chabot_smr <- slope_df_filter_2 %>%
    dplyr::filter(phase == "smr")

# Extract distinct IDs
ids <- labchart_chabot_smr %>%
    dplyr::distinct(id) %>%
    dplyr::pull()

# Initialise an empty list to store SMR data
smr_list <- list()

# Process each ID
for (id_i in ids) {
    tryCatch({
        # Filter the data for the current ID
        df_i <- labchart_chabot_smr %>%
            dplyr::filter(id == id_i) %>%
            dplyr::mutate(abs_mo2corr = abs(mo2corr))

        # Calculate SMR results
        calcSMR_results <- calcSMR(df_i$abs_mo2corr)
        CVmlnd_i <- calcSMR_results$CVmlnd
        quant_i <- calcSMR_results$quant %>%
            as_tibble()
        quant_20per_i <- quant_i$value[3]
        mlnd_i <- calcSMR_results$mlnd
        smr_value <- if_else(CVmlnd_i < 5.4, mlnd_i, quant_20per_i)
        smr_type <- if_else(CVmlnd_i < 5.4, "mlnd", "quant_20per")
        smr_value <- if_else(is.na(smr_value), quant_20per_i, smr_value)
        smr_type <- if_else(is.na(smr_type), "quant_20per", smr_type)

        # Create a data frame for the current ID
        smr_df <- tibble(id = id_i, smr = smr_value, smr_est = smr_type)

    }, error = function(e) {
        # Handle errors by assigning NA values
        smr_df <- tibble(id = id_i, smr = NA, smr_est = NA)
    })

    # Append to the list
    smr_list[[id_i]] <- smr_df
}

# Combine all individual SMR data frames into one
smr_df <- bind_rows(smr_list) %>%
    dplyr::rename(smr_chabot = smr, smr_chabot_method = smr_est)

slope_df_filter_3 <- slope_df_filter_2 %>%
    dplyr::left_join(., smr_df, by = "id")

8.6 Transforming MO₂

Here we are transforming the MO₂ units. The resulting values are as follows:

  • MO2 is absolute value of the background and leak corrected MO₂ slope from Labchart (mo2corr) times the net volume of the chamber (volume - fish mass), × 60, × 60, to achieve MO₂ as mg-1 O₂ h-1

  • MO2_g is MO2 divided by fish mass to achieve MO₂ as mg-1 O₂ g-1 h-1 (i.e. mass standardised)

  • SMR absolute value of the SMR estimates using methods described by Chabot, Steffensen and Farrell (2016)[1] times the net volume of the chamber (volume - fish mass), × 60, × 60, to achieve SMR as mg-1 O₂ g-1 h-1)

  • SMR_g is SMR divided by fish mass to achieve SMR as mg-1 O₂ g-1 h-1 (i.e. mass standardised)

  • DO is dissolved oxygen percentage calculated from O₂ values (mg-1 L-1) using the recorded temperature, salinity, and a constant atmospheric pressure (Pa; 1013.25)

  • o2_kpa is the O₂ concentration in kilopascal (kpa). This is used to make a comparative figure only.

# Combine back into one data frame
slope_tidy <- slope_df_filter_3 %>% 
    dplyr::mutate(DO = conv_o2(
                   o2 = o2,
                   from = "mg_per_l",
                   to = "percent_a.s.",
                   temp = temp, #C
                   sal = measured_salinity,
                   atm_pres = 1013.25),
                  o2_kpa = conv_o2(
                   o2 = o2,
                   from = "mg_per_l",
                   to = "kPa",
                   temp = temp, #C
                   sal = measured_salinity,
                   atm_pres = 1013.25),
                  net_volume = volume - mass/1000,
                  MO2 = abs(mo2corr)*net_volume*60*60,
                  MO2_g = MO2/mass,
                  SMR = abs(smr_chabot)*net_volume*60*60,
                  SMR_g = SMR/mass
                  )

9 📊 Visualising data

9.1 O₂ vs MO₂

9.1.1 Figure S1

This interactive was used to identify any outliers, or potential errors.

lm_lines <- slope_tidy %>%
  group_by(id) %>%
  summarise(
    DO_seq = list(seq(min(DO), max(DO), length.out = 100)),  # Generate a sequence of DO values
    MO2_pred = list(predict(lm(MO2_g ~ DO, data = cur_data()), newdata = data.frame(DO = seq(min(DO), max(DO), length.out = 100))))
  ) %>%
  unnest(c(DO_seq, MO2_pred))  # Expand lists into rows

# Create scatter plot with markers for each fish
p <- plot_ly(
  data = slope_tidy,
  x = ~DO,
  y = ~MO2_g,
  type = "scatter",
  mode = "markers",
  color = ~id,  # Colour points by fish ID
  marker = list(opacity = 0.6),
  name = ~id
)

# Add regression lines for each fish
p <- p %>%
  add_trace(
    data = lm_lines,
    x = ~DO_seq,
    y = ~MO2_pred,
    type = "scatter",
    mode = "lines",
    color = ~id,  # Ensure each line matches its corresponding fish
    line = list(width = 1, dash = "solid"),
    showlegend = FALSE  # Avoid cluttering the legend
  )

# Final layout
p <- p %>%
  layout(
    title = "MO<sub>2</sub> vs Dissolved Oxygen with individual linear regressions",
    xaxis = list(title = "Dissolved Oxygen (%)"),
    yaxis = list(title = "MO<sub>2</sub> (mg<sup>-1</sup> O<sub>2</sub> g<sup>-1</sup> h<sup>-1</sup>)"),
    showlegend = FALSE
  )

# Display plot
p

Figure S1: interactive plot of metabolic rate measurements (MO₂; mg O₂ g-1h-1) by dissolved oxygen percentage (DO) for all fish, including all estimates during the SMR phase (i.e. intermittent phase). Individual linear regression were fitted for visual reference, and do not represent the best fitting regression.

9.1.2 Figure S2

Looking at the difference responses in the two salinity groups.

salinity_summary <- slope_tidy %>% 
  dplyr::group_by(salinity_group) %>% 
  dplyr::reframe(n = length(unique(id)))

slope_tidy %>% 
  ggplot(aes(y = MO2_g, x = DO, colour = id)) + # Default aesthetics
  geom_point(show.legend = FALSE) +
  geom_smooth(aes(group = id), method = "lm", se = FALSE, colour = scales::alpha("black", 0.5)) + # Transparent black lines
  geom_smooth(method = "lm", se = TRUE, colour = "red") + # Overall smooth line
  geom_smooth(method = "gam", formula = y ~ s(x, bs = "cs"), se = TRUE, colour = "red", linetype = "dashed") +
  theme_clean() +
  facet_wrap(~salinity_group) +
  labs(
    subtitle = "mo2 vs o2 by salinity treatment",
    x = "Dissolved oxygen percentage (DO)",
    y = "MO2 (O2 mg/g/h)"
  ) +
  geom_text(data = salinity_summary,
            aes(x = -Inf, y = Inf, label = paste0("italic(n) == ", n)),
            hjust = -0.1, vjust = 1.2, inherit.aes = FALSE, parse = TRUE)

Figure S2: Metabolic rate measurements (MO₂; mg O₂ g-1h-1) by dissolved oxygen percentage (DO) for fish from the two salinity treatments, including all estimates during the SMR phase (i.e. intermittent phase). Individual linear regression were fitted for visual reference, and do not represent the best fitting regression. The solid red and dashed red line represents the global trend in each of the treatments, both a linear (solid red) and non-linear (dashed red; Generalized Additive Model fitted with geom_smooth).

9.1.3 Figure S3

A plot to look at the different chamber types

chamber_summary <- slope_tidy %>% 
  dplyr::group_by(chamber_type) %>% 
  dplyr::reframe(n = length(unique(id)))

slope_tidy %>% 
  ggplot(aes(y = MO2_g, x = DO, colour = id)) + # Default aesthetics
  geom_point(show.legend = FALSE) +
  geom_smooth(aes(group = id), method = "lm", se = FALSE, colour = scales::alpha("black", 0.5)) + # Transparent black lines
  geom_smooth(method = "lm", se = TRUE, colour = "red") + # Overall smooth line
  geom_smooth(se = TRUE, colour = "red", linetype = "dashed") +
  theme_clean() +
  facet_wrap(~chamber_type, scale = "free") +
  labs(
    subtitle = "mo2 vs o2 by chamber type",
    x = "Dissolved oxygen percentage (DO)",
    y = "MO2 (mg O2 g/h)"
  ) +
  geom_text(data = chamber_summary,
            aes(x = -Inf, y = Inf, label = paste0("italic(n) == ", n)),
            hjust = -0.1, vjust = 1.2, inherit.aes = FALSE, parse = TRUE)

Figure S3: Metabolic rate measurements (MO₂; mg O₂ g-1h-1) by dissolved oxygen percentage (DO) for fish tested in the 4 different chamber types, including all estimates during the SMR phase (i.e. intermittent phase). Individual linear regression were fitted for visual reference, and do not represent the best fitting regression. The solid red and dashed red line represents the global trend in each of the treatments, both a linear (solid red) and non-linear (dashed red; Generalized Additive Model fitted with geom_smooth).

9.1.4 Figure S4

Here, we have recreated a similar plot to that presented in Urbina, Glover, and Forster (2012)[1] and have extracted the mean level data from Figure 1a using the metaDigitise package in R[3]. This data is called urbina_et_al_2012. This allows us to compare the differences in Mo2 and the relationship between Mo2 and O2.

First making a binned data frame to match Urbina, Glover, and Forster (2012) as closely as possible.

min_o2_kpa <- min(slope_tidy$o2_kpa, na.rm = TRUE)
max_o2_kpa <- max(slope_tidy$o2_kpa, na.rm = TRUE)

time_bin_df <- slope_tidy %>%
  mutate(o2_group = cut(o2_kpa, 
                        breaks = seq(min_o2_kpa, max_o2_kpa, length.out = 13), # 11 intervals, so 12 breakpoints
                        labels = paste0("Group ", 1:12), 
                        include.lowest = TRUE)) %>% 
  dplyr::group_by(o2_group) %>% 
  dplyr::reframe(mean_MO2_g = mean(MO2_g)*31.25,
                 mean_o2_kpa = mean(o2_kpa),
                 n = length(MO2_g)*31.25,
                 MO2_g_sd = sd(MO2_g)*31.25,
                 o2_kpa_sd = sd(o2_kpa))

Now the plot with our mean data and the mean data from Urbina, Glover, and Forster (2012)[1]

n <- slope_tidy %>%
    dplyr::distinct(id) %>%
    nrow(.)

# Existing plot
p <- time_bin_df %>%
    ggplot() + geom_point(data = slope_tidy, aes(y = MO2_g * 31.25, x = o2_kpa),
    size = 2, color = "grey", alpha = 0.3) + geom_point(data = time_bin_df, aes(y = mean_MO2_g,
    x = mean_o2_kpa), size = 3, colour = "#0E4C92", show.legend = FALSE) + geom_errorbar(data = time_bin_df,
    aes(y = mean_MO2_g, x = mean_o2_kpa, ymin = mean_MO2_g - MO2_g_sd, ymax = mean_MO2_g +
        MO2_g_sd), width = 0.15, colour = "#0E4C92") + geom_errorbarh(data = time_bin_df,
    aes(y = mean_MO2_g, x = mean_o2_kpa, xmin = mean_o2_kpa - o2_kpa_sd, xmax = mean_o2_kpa +
        o2_kpa_sd), height = 0.4, colour = "#0E4C92") + geom_point(data = urbina_et_al_2012,
    aes(x = o2_mean, y = mo2_mean), size = 3, shape = 21, fill = "#D21F3C", color = "#D21F3C",
    stroke = 1) + geom_errorbar(data = urbina_et_al_2012, aes(x = o2_mean, ymin = mo2_mean -
    mo2_sd, ymax = mo2_mean + mo2_sd), width = 0.2, colour = "#D21F3C") + geom_errorbarh(data = urbina_et_al_2012,
    aes(y = mo2_mean, xmin = o2_mean - o2_sd, xmax = o2_mean + o2_sd), height = 0.4,
    colour = "#D21F3C") + annotate("text", x = 0, y = 16, label = bquote(atop("Current data (blue), " *
    italic(n) * " = " * .(n), "Urbina data (red), " * italic(n) * " = 67")), hjust = 0,
    vjust = 1, size = 4) + theme_clean() + labs(subtitle = "", x = "PO2 (kPa)", y = "MO2 (umol O2 g/h)") +
    scale_y_continuous(limits = c(0, 16), breaks = seq(0, 16, by = 2)) + scale_x_continuous(limits = c(0,
    22), breaks = seq(0, 22, by = 2))

p

Figure S4: Mean and standard error of metabolic rate (MO2 μmol O2 g-1 h-1) and oxygen concentration (PO2 kPa) using 12 evenly spaced bins over the O2 range of observed data (blue dots). Compared against the mean and standard error reported in Urbina (2012)[1] (red dots). The grey dots are the raw observed data form the present study.

9.2 Routine MO₂

Making an SMR phase only data frame

slope_tidy_smr <- slope_tidy %>%
    dplyr::filter(phase == "smr")

9.2.1 Figure S5

Routine MO2 by salinity

mean_mo2_salinity <- slope_tidy_smr %>% 
  dplyr::group_by(salinity_group) %>% 
  dplyr::reframe(mean_mo2 = mean(MO2, na.rm = TRUE))

fig_i <- ggplot() +
    geom_violin(data = slope_tidy_smr, aes(x = salinity_group, y = MO2, fill = salinity_group), color = NA, alpha = 0.3) +
  geom_jitter(data = slope_tidy_smr, aes(x = salinity_group, y = MO2, fill = salinity_group),
                       shape = 21, size = 2, color = "black", alpha = 0.2) +
  geom_boxplot(data = slope_tidy_smr, aes(x = salinity_group, y = MO2, fill = salinity_group),
                        size = 1, alpha = 0.5, outlier.shape = NA, width = 0.3) +
  geom_point(data = mean_mo2_salinity, 
                aes(x = salinity_group, y = mean_mo2, fill = salinity_group), 
                size = 3, alpha = 0.8, colour = "black", stroke = 2) +
  scale_fill_manual(values = c("#4B5320", "#000080")) +  # Custom fill colours
  scale_colour_manual(values = c("#4B5320", "#000080")) +
  theme_clean() +
  theme(legend.position = "none") +
  labs(
    subtitle = "",
    x = "Salinity group (ppt)",
    y = "Routine MO2 (mg O2 g/h)"
  )

fig_i

Figure S5: Plot of all MO2 measures during SMR phase by salinity treatment. The small points are the raw observed values, the shaded area behind the points is the a kernel density of the observed data, the box plot shows the median and interquartile range (IQR), and the large point shows the mean.

9.3 SMR

9.3.1 Figure S6

Here’s the same plot but for only the SMR, as estimated with calcSMR function by Chabot, Steffensen and Farrell (2016)[1]. Specifically, we use mean of the lowest normal distribution (MLND) where CVmlnd < 5.4 and the mean of the lower 20% quantile (q0.2) were CVmlnd > 5.4. If CVmlnd is not calculated we have used q0.2.

smr_only <- slope_tidy_smr %>% 
  dplyr::group_by(id) %>% 
  dplyr::slice(1)

mean_smr_salinity <- smr_only %>% 
  dplyr::group_by(salinity_group) %>% 
  dplyr::reframe(mean_smr = mean(SMR, na.rm = TRUE))

fig_i <- ggplot() +
    geom_violin(data = smr_only, aes(x = salinity_group, y = SMR, fill = salinity_group), color = NA, alpha = 0.3) +
  geom_jitter(data = smr_only, aes(x = salinity_group, y = SMR, fill = salinity_group),
                       shape = 21, size = 2, color = "black", alpha = 0.2) +
  geom_boxplot(data = smr_only, aes(x = salinity_group, y = SMR, fill = salinity_group),
                        size = 1, alpha = 0.5, outlier.shape = NA, width = 0.3) +
  geom_point(data = mean_smr_salinity, 
                aes(x = salinity_group, y = mean_smr, fill = salinity_group), 
                size = 3, alpha = 0.8, colour = "black", stroke = 2) +
  scale_fill_manual(values = c("#4B5320", "#000080")) +  # Custom fill colours
  scale_colour_manual(values = c("#4B5320", "#000080")) +
  theme_clean() +
  theme(legend.position = "none") +
  labs(
    subtitle = "",
    x = "Salinity group (ppt)",
    y = "MO2 (mg O2 g/h)"
  )

fig_i

Figure S6: The standard metabolic rate (SMR) by salinity treatment. The small points are the raw observed values, the shaded area behind the points is the a kernel density of the observed data, the box plot shows the median and interquartile range (IQR), and the large point shows the mean.

9.4 Individual O₂, MO₂, and SMR

Here we will plot the individual relationship between O₂, MO₂, and SMR for all fish

Create output directory if needed

9.4.1 Figure S7

Loop through each fish ID to create a plot, save these to a single PDF file

library(ggplot2)
library(dplyr)

# Define consistent colors for all phases
phase_colors <- c(
  "50c"  = "#e56b6f",
  "75c"  = "#b56576",
  "100c" = "#6d597a",
  "smr"  = "#355070"
)

ids <- slope_tidy %>% 
  dplyr::distinct(id) %>% 
  pull(id) %>% 
  as.list()

MO2_plot_list <- list()

# 1) Open the PDF device once
pdf(
  file   = file.path(output_fig_slopes_wd, "combined_slopes.pdf"), 
  width  = 8, 
  height = 6
)

# 2) Loop over IDs and create each plot
for (id_i in ids) {
  
  smr <- slope_tidy %>% 
    dplyr::filter(id == id_i) %>% 
    dplyr::slice(1) %>% 
    dplyr::pull(SMR)
  
  plot <- slope_tidy %>% 
    dplyr::filter(id == id_i) %>% 
    ggplot(aes(x = o2, y = MO2)) +
    geom_hline(yintercept = smr, linetype = "dashed", color = "darkred") +
    geom_point(aes(colour = phase)) +
    scale_color_manual(values = phase_colors, drop = FALSE) +  # Ensures consistent colours
    theme_clean() +
    labs(
      subtitle = paste0(id_i, " slopes"),
      x = "Mean o2 (mg_per_l)",
      y = "abs(mo2) (mg_per_l)"
    )
  
  # Instead of saving each plot separately, just print it
  print(plot)
  
  MO2_plot_list[[id_i]] <- plot
}

# 3) Close the PDF device *after* the loop
dev.off()
## png 
##   2
for (p in MO2_plot_list) {
  print(p)
}

Figure S7: Metabolic rate measurements (MO₂; mg O₂ g-1h-1 by O2) at each of the four experimental phase (the over night SMR intermittent-flow respirometry phase, closed phases at 50%, 75% or 100% O2). The estimated SMR value is represented as a dashed red line.

10 🧮 Analysis

10.1 Routine MO2

10.1.1 Scaling predictors

Here we scale our predictors for the model

scale_list <- c("temp", "order", "mass")

slope_tidy_smr <- slope_tidy_smr %>%
    dplyr::mutate(across(all_of(scale_list), ~scale(.x, center = TRUE, scale = FALSE),
        .names = "{.col}_z"), light_dark_c = if_else(light_dark == "light", 0.5,
        -0.5))

10.1.2 Model structure

Here we will use a Bayesian Generalised Linear Mixed Model (GLMM) with a Gamma distribution and a log link, where the shape parameter (K) is also modelled as a function of predictors. This models MO2 by salinity during the SMR phase to see if the fish held at different salinities have different SMRs. We have also added a few scaled predictors, that may help describe variation in the data, such as mass (g; 0.21–1.6) temperature (°C; 13.841–14.277), measurement order (1–28), and light/dark cycle (light or dark; light between 07:00:00 and 19:00:00), we also include a random effect for fish id to account for multiple MO2 measures on each fish (1 - 58). We allowed the the shape parameter (K) to vary as a function of some of the predictors (e.g. salinity_group, order_z) to improve fit.

mo2_gamma_bf <- bf(MO2 ~ temp_z + order_z + light_dark_c + mass_z + salinity_group +
    (1 | id), shape ~ salinity_group + order_z, family = Gamma(link = "log"))

10.1.3 Prior selection

These are the default priors. We will use these.

default_prior <- get_prior(mo2_gamma_bf, data = slope_tidy_smr, family = Gamma(link = "log"))
default_prior %>%
    gt()
prior class coef group resp dpar nlpar lb ub source
b default
b light_dark_c default
b mass_z default
b order_z default
b salinity_group9 default
b temp_z default
student_t(3, -2.7, 2.5) Intercept default
student_t(3, 0, 2.5) sd 0 default
sd id default
sd Intercept id default
b shape default
b order_z shape default
b salinity_group9 shape default
student_t(3, 0, 2.5) Intercept shape default

10.1.4 Run model

Here we run the model, I have hashed this out because I have saved the model for quick reloading.

⏭️ Skip this step if you have downloaded the saved model ‘mo2_mod_gamma.rds’

Here we reload the model

setwd(output_mods_wd)

mo2_mod_gamma <- readRDS(file = "mo2_mod_gamma.rds")

10.1.5 Model diagnostics

Checking model convergence

color_scheme_set("red")
plot(mo2_mod_gamma, ask = F)

Checking rhat are equal to one

summary(mo2_mod_gamma)
##  Family: gamma 
##   Links: mu = log; shape = log 
## Formula: MO2 ~ temp_z + order_z + light_dark_c + mass_z + salinity_group + (1 | id) 
##          shape ~ salinity_group + order_z
##    Data: slope_tidy_smr (Number of observations: 893) 
##   Draws: 4 chains, each with iter = 8000; warmup = 1000; thin = 2;
##          total post-warmup draws = 14000
## 
## Multilevel Hyperparameters:
## ~id (Number of levels: 58) 
##               Estimate Est.Error l-95% CI u-95% CI Rhat Bulk_ESS Tail_ESS
## sd(Intercept)     0.34      0.03     0.28     0.41 1.00     3864     5351
## 
## Regression Coefficients:
##                       Estimate Est.Error l-95% CI u-95% CI Rhat Bulk_ESS
## Intercept                -2.55      0.06    -2.68    -2.43 1.00     2450
## shape_Intercept           2.57      0.07     2.43     2.70 1.00    12540
## temp_z                   -0.36      0.22    -0.80     0.08 1.00     6496
## order_z                   0.00      0.00    -0.00     0.01 1.00    11555
## light_dark_c              0.14      0.02     0.09     0.19 1.00    13253
## mass_z                    1.24      0.17     0.90     1.58 1.00     3339
## salinity_group9          -0.12      0.09    -0.30     0.07 1.00     2440
## shape_salinity_group9     0.11      0.10    -0.09     0.30 1.00    12382
## shape_order_z            -0.04      0.01    -0.06    -0.02 1.00    12079
##                       Tail_ESS
## Intercept                 4218
## shape_Intercept          12262
## temp_z                    9064
## order_z                  12406
## light_dark_c             12316
## mass_z                    5807
## salinity_group9           4288
## shape_salinity_group9    12635
## shape_order_z            12068
## 
## Draws were sampled using sampling(NUTS). For each parameter, Bulk_ESS
## and Tail_ESS are effective sample size measures, and Rhat is the potential
## scale reduction factor on split chains (at convergence, Rhat = 1).

Using leave one out (loo) measure of fit, the model appears to preform well, all Pareto k estimates are good (k < 0.7)

loo(mo2_mod_gamma)
## 
## Computed from 14000 by 893 log-likelihood matrix.
## 
##          Estimate   SE
## elpd_loo   2251.6 39.0
## p_loo        71.5  7.0
## looic     -4503.3 77.9
## ------
## MCSE of elpd_loo is 0.1.
## MCSE and ESS estimates assume MCMC draws (r_eff in [0.7, 1.0]).
## 
## All Pareto k estimates are good (k < 0.7).
## See help('pareto-k-diagnostic') for details.

Model predictions generally align with the observed data

color_scheme_set("red")
p <- pp_check(mo2_mod_gamma, type = "dens_overlay")
p

10.1.6 📈 Results

We did not see a meaningful difference between the routine metabolic rate for fish from the two salinity treatments.

10.1.6.1 Table S1

Table S1: Fixed effect Estimates (β) and 95% Credible Intervals (95% CI)

model_est <- fixef(mo2_mod_gamma, probs = c(0.025, 0.975)) %>%
    as.data.frame() %>%
    tibble::rownames_to_column(var = "Predictor") %>%
    dplyr::mutate(β = round(Estimate, 3), Q2.5 = round(Q2.5, 3), Q97.5 = round(Q97.5,
        3), `95% CI` = paste0("[", Q2.5, ", ", Q97.5, "]"))

model_est %>%
    dplyr::select(Predictor, "β", "95% CI") %>%
    gt()
Predictor β 95% CI
Intercept -2.554 [-2.68, -2.427]
shape_Intercept 2.570 [2.434, 2.704]
temp_z -0.360 [-0.804, 0.077]
order_z 0.002 [-0.002, 0.005]
light_dark_c 0.137 [0.089, 0.186]
mass_z 1.241 [0.903, 1.585]
salinity_group9 -0.115 [-0.301, 0.067]
shape_salinity_group9 0.107 [-0.087, 0.304]
shape_order_z -0.040 [-0.059, -0.022]

Looking at the marginal mean difference between salinity groups

em_results <- emmeans(mo2_mod_gamma, ~salinity_group)
contrast_results <- contrast(em_results, method = "pairwise")
em_results_df <- em_results %>%
    tidy() %>%
    mutate(across(where(is.numeric), ~exp(.)))
contrast_results_df <- contrast_results %>%
    tidy() %>%
    mutate(across(where(is.numeric), ~exp(.)))


em_results_df %>%
    gt()
salinity_group estimate lower.HPD upper.HPD
0 0.07775342 0.06851859 0.08822464
9 0.06929211 0.06056439 0.07873088

Pulling the emmeans draws for our plot

emmeans_draws <- mo2_mod_gamma %>%
    emmeans(~salinity_group) %>%
    gather_emmeans_draws() %>%
    dplyr::mutate(.value = exp(.value), salinity_group = as.character(salinity_group))

emmeans_contrast_draws <- mo2_mod_gamma %>%
    emmeans(~salinity_group) %>%
    contrast(method = "pairwise") %>%
    gather_emmeans_draws() %>%
    dplyr::mutate(.value = exp(.value))

10.1.6.2 Figure S8

NOTE: This plot is in the main text of the manuscript as Figure 1a

mean_mo2_salinity <- slope_tidy_smr %>% 
  dplyr::group_by(salinity_group) %>% 
  dplyr::reframe(mean_mo2 = mean(MO2, na.rm = TRUE))


fig_1 <- ggplot() +
  geom_violin(data = slope_tidy_smr,
              aes(x = salinity_group, y = MO2, fill = salinity_group),
              color = NA, alpha = 0.2) +
  geom_jitter(data = slope_tidy_smr,
              aes(x = salinity_group, y = MO2, fill = salinity_group),
              shape = 21, width = 0.3, size = 1, color = "black", alpha = 0.1) +
    geom_point(data = mean_mo2_salinity,
             aes(x = salinity_group, y = mean_mo2, fill = salinity_group),
             size = 4, alpha = 1, stroke = 2, color = "black", shape = 21,
              position = position_nudge(x = 0.05)) +
  stat_pointinterval(data = emmeans_draws, 
                     aes(x = salinity_group, y = .value),
                     color = "black", fill = "grey", point_interval = "mean_qi", .width = 0.95, shape = 21,  stroke = 2, point_size = 4, alpha = 1,
                     position = position_nudge(x = -0.05)) +
  scale_fill_manual(values = c("#4B5320", "#000080")) +  # Custom fill colours
  scale_colour_manual(values = c("#4B5320", "#000080")) +
  theme_clean() +
  theme(legend.position = "none") +
  labs(
    subtitle = "",
    x = "Salinity group (ppt)",
    y = "Routine MO2 (mg O2 g/h)"
  )

fig_1

Figure 1a: Routine metabolic rate (i.e. MO2 (mg-1 O2 h-1) measured during SMR meassurments) plotted by salinity treatment. The small transparent points are the observed values, the shaded area behind the points is the a kernel density of the observed data, the large coloured point (to the right) is the observed mean, the large grey point with error bars (to the left) is the model estimated marginal mean (eemean) 95% Credible Intervals (95% CI).

10.2 SMR

10.2.1 Formating and scaling data

Here we are filtering the data frame to have only measure per fish for the SMR estimate

scale_list <- c("temp_mean", "mass", "cycles")

smr <- slope_tidy_smr %>%
    dplyr::group_by(id) %>%
    dplyr::reframe(temp_mean = mean(temp), mass = mass[1], SMR = SMR[1], salinity_group = salinity_group[1],
        cycles = length(order)) %>%
    dplyr::mutate(across(all_of(scale_list), ~scale(.x, center = TRUE, scale = FALSE),
        .names = "{.col}_z"))

10.2.2 Model structure

Here we will use a Bayesian Generalised Linear Mixed Model (GLMM) with a Gamma distribution and a log link Gamma(link = "log"), where the shape parameter (K) is also modelled as a function of the salinity group, shape ~ salinity_group. This models MO2 by salinity during the SMR phase to see if the fish held at different salinities have different SMRs. We have also added a few scaled predictors, that may help describe variation in the data, such as mass (g; 0.21–1.6) temperature (°C; 13.841–14.277), measurement order (1–28), and light/dark cycle (light or dark; light between 07:00:00 and 19:00:00), we also include a random effect for fish id to account for multiple MO2 measures on each fish (1 - 58). We allowed the the shape parameter (K) to vary as a function of some of the predictors (e.g. salinity_group, order_z) to improve fit.

smr_gamma_bf <- bf(SMR ~ temp_mean_z + cycles_z + mass_z + salinity_group, shape ~
    salinity_group, family = Gamma(link = "log"))

10.2.3 Prior selection

These are the default priors for the model. We will use these.

priors_default <- get_prior(smr_gamma_bf, data = smr, family = Gamma(link = "log"))
priors_default %>%
    gt()
prior class coef group resp dpar nlpar lb ub source
b default
b cycles_z default
b mass_z default
b salinity_group9 default
b temp_mean_z default
student_t(3, -2.8, 2.5) Intercept default
b shape default
b salinity_group9 shape default
student_t(3, 0, 2.5) Intercept shape default

10.2.4 Run model

Here we run the model, I have hashed this out because I have saved the model for quick reloading.

⏭️ Skip this step if you have downloaded the saved model ‘smr_mod_gamma.rds’

Here we reload the model

setwd(output_mods_wd)

smr_mod_gamma <- readRDS(file = "smr_mod_gamma.rds")

10.2.5 Model diagnostics

Checking model convergence

color_scheme_set("red")
plot(smr_mod_gamma, ask = F)

Using leave one out (loo) measure of fit, the model appears to preform well, one Pareto k estimates falls outside the good range (0.7, 1]

loo(smr_mod_gamma)
## 
## Computed from 14000 by 58 log-likelihood matrix.
## 
##          Estimate   SE
## elpd_loo    134.4  9.4
## p_loo         9.5  3.5
## looic      -268.9 18.9
## ------
## MCSE of elpd_loo is NA.
## MCSE and ESS estimates assume MCMC draws (r_eff in [0.8, 0.9]).
## 
## Pareto k diagnostic values:
##                          Count Pct.    Min. ESS
## (-Inf, 0.7]   (good)     56    96.6%   297     
##    (0.7, 1]   (bad)       2     3.4%   <NA>    
##    (1, Inf)   (very bad)  0     0.0%   <NA>    
## See help('pareto-k-diagnostic') for details.

Model predictions generally align with the observed data, but there is a lot of uncertainty.

color_scheme_set("red")
p <- pp_check(smr_mod_gamma, type = "dens_overlay")
p

10.2.6 📈 Results

We did not see a meaningful difference between the routine metabolic rate for fish from the two salinity treatments.

10.2.6.1 Table S2

Table S2: Fixed effect Estimates (β) and 95% Credible Intervals (95% CI) from a Bayesian Generalised Linear Mixed Model (GLMM) with a Gamma distribution and a log link

model_est <- fixef(smr_mod_gamma, probs = c(0.025, 0.975)) %>%
    as.data.frame() %>%
    tibble::rownames_to_column(var = "Predictor") %>%
    dplyr::mutate(β = round(Estimate, 3), Q2.5 = round(Q2.5, 3), Q97.5 = round(Q97.5,
        3), `95% CI` = paste0("[", Q2.5, ", ", Q97.5, "]"))

model_est %>%
    dplyr::select(Predictor, "β", "95% CI") %>%
    gt()
Predictor β 95% CI
Intercept -2.751 [-2.889, -2.615]
shape_Intercept 2.018 [1.435, 2.545]
temp_mean_z -0.364 [-1.548, 0.802]
cycles_z -0.003 [-0.027, 0.019]
mass_z 1.379 [0.947, 1.818]
salinity_group9 -0.096 [-0.302, 0.11]
shape_salinity_group9 -0.081 [-0.913, 0.756]

Looking at the marginal mean difference between salinity groups

em_results <- emmeans(smr_mod_gamma, ~salinity_group)
contrast_results <- contrast(em_results, method = "pairwise")
em_results_df <- em_results %>%
    tidy() %>%
    mutate(across(where(is.numeric), ~exp(.)))
contrast_results_df <- contrast_results %>%
    tidy() %>%
    mutate(across(where(is.numeric), ~exp(.)))


em_results_df %>%
    gt()
salinity_group estimate lower.HPD upper.HPD
0 0.06382226 0.05568229 0.07322901
9 0.05801414 0.04982011 0.06710878

Pulling the emmeans draws for our plot

emmeans_draws <- smr_mod_gamma %>%
    emmeans(~salinity_group) %>%
    gather_emmeans_draws() %>%
    dplyr::mutate(.value = exp(.value), salinity_group = as.character(salinity_group))

emmeans_contrast_draws <- smr_mod_gamma %>%
    emmeans(~salinity_group) %>%
    contrast(method = "pairwise") %>%
    gather_emmeans_draws() %>%
    dplyr::mutate(.value = exp(.value))

10.2.6.2 Figure S9

This plot is in the main text of the manuscript as Figure 1b

mean_smr_salinity <- smr %>%
  dplyr::group_by(salinity_group) %>%
  dplyr::reframe(mean_SMR = mean(SMR, na.rm = TRUE))


fig_1b <- ggplot() +
  geom_violin(data = smr,
              aes(x = salinity_group, y = SMR, fill = salinity_group),
              color = NA, alpha = 0.2) +
  geom_jitter(data = smr,
              aes(x = salinity_group, y = SMR, fill = salinity_group),
              shape = 21, width = 0.3, size = 1, color = "black", alpha = 0.1) +
    geom_point(data = mean_smr_salinity,
             aes(x = salinity_group, y = mean_SMR, fill = salinity_group),
             size = 4, alpha = 1, stroke = 2, color = "black", shape = 21,
              position = position_nudge(x = 0.05)) +
  stat_pointinterval(data = emmeans_draws,
                     aes(x = salinity_group, y = .value),
                     color = "black", fill = "grey", point_interval = "mean_qi", .width = 0.95, shape = 21,  stroke = 2, point_size = 4, alpha = 1,
                     position = position_nudge(x = -0.05)) +
  scale_fill_manual(values = c("#4B5320", "#000080")) +  # Custom fill colours
  scale_colour_manual(values = c("#4B5320", "#000080")) +
  theme_clean() +
  theme(legend.position = "none") +
  labs(
    subtitle = "",
    x = "Salinity group (ppt)",
    y = "Standard metabolic rate (SMR; mg O2 g/h)"
  )

fig_1b

Figure 1b: The standard metabolic rate estimate (mg-1 O2 h-1) plotted by salinity treatment. The small transparent points are the observed values, the shaded area behind the points is the a kernel density of the observed data, the large coloured point (to the right) is the observed mean, the large grey point with error bars (to the left) is the model estimated marginal mean (eemean) 95% Credible Intervals (95% CI).

10.3 Incremental regression analyses

Here we are following the methods Urbina et al. (2012)[1] with an incremental regression analyses, in order to determine the best fit for the MO2 vs O2 data

This analysis approach evaluates the relative ‘fit’ of each polynomial order equation starting at zero and increasing to the third order, permitting a mathematical assessment of whether fish were oxyconforming or oxyregulatoring. If the data is best fitted/predicted by a single linear relationship (1st-order polynomial) with a positive slope, this would suggests the fish were oxyconforming. Alternately, if the relationship is best modelled by a flat regression (0th-order polynomial), or a higher order polynomial (2nd or 3rd-order polynomial) the fish is likely oxyregulatoring.

output_mods_bayes_wd <- paste0(output_mods_wd, "./bayes-regs")
if (!dir.exists(output_mods_bayes_wd)) {
    dir.create(output_mods_bayes_wd)
}

10.3.1 Building Bayesian regressions

Here we are using a Bayesian approach to model fitting with brm. These models take a long time to run, so I have saved them and re-loaded them to save time. I have also saved the summary data produced from the models, to save time, you can simply skip the hashed code and input the resulting summary data.

We will run our custom function, bayes_incremental_regression_by_id.

⏭️ Skip this step if you have already run this once, or have downloaded the saved models or saved datafiles from GitHub (that’s why its hashed out). This code takes a while to run

I have fund that I get an error, ‘Caused by error in socketConnection():’. I think the system may be hitting a limit on the number of simultaneous socket connections. If you try and run this code and also get this issue, try reducing the number of parallel workers: plan(multisession, workers = 2).

# ids <- slope_tidy %>% dplyr::distinct(id) %>% pull(id) plan(multisession)
# future_map( ids, bayes_incremental_regression_by_id, id_name = 'id', data =
# slope_tidy, response = 'MO2_g', predictor = 'DO', seed_number = 143019,
# save_models = TRUE, mod_output_wd = output_mods_bayes_wd ) plan(sequential)

Load all models and store in a list, will use a lot of memory. You can also skip this step and load the resulting data frames below. I am using the custom function load_rds, so we can compare them and generate predictions.

⏭️ Skip this step if you have downloaded the saved data pulled from these models have also hased this out, bayes_reg_mods_fit.csv’ and ‘bayes_reg_mods_predictions.csv’. These are in the mod-data directory.

# bayes_reg_mods <- load_rds(model_dw = output_mods_bayes_wd)

10.3.2 Model fits

⏭️ Skip this step if you have downloaded 💿 bayes_reg_mods_fit.csv.

It gets model fit parameters loo and r2 using the custom function, incremental_regression_bayes_fits. This code takes a while to run

# setwd(mod_data_wd) bayes_reg_mods_fit <-
# incremental_regression_bayes_fits(models = bayes_reg_mods)
# write.csv(bayes_reg_mods_fit, 'bayes_reg_mods_fit.csv', row.names = FALSE)

Reading in this model fit data frame, in the case you did not load in all the models.

setwd(mod_data_wd)
bayes_reg_mods_fit <- read.csv("bayes_reg_mods_fit.csv")

Selecting the best fitting model

elpd_loo, or the expected log pointwise predictive density for leave-one-out cross-validation, is a metric used in Bayesian model evaluation to assess the predictive accuracy of a model. The elpd_loo is an approximation of how well the model is expected to predict new data, based on leave-one-out cross-validation. Higher elpd_loo values indicate better predictive performance.

best_fit_bayes_reg <- bayes_reg_mods_fit %>%
    dplyr::group_by(id) %>%
    dplyr::mutate(elpd_loo_rank = rank(-elpd_loo)) %>%
    dplyr::select(id, model_type, elpd_loo, r2, elpd_loo_rank, r2_q2.5, r2_q97.5,
        estimate_DO, conf.low_DO, conf.high_DO) %>%
    ungroup()

10.3.3 Model predictions

⏭️ Skip this step if you have downloaded 💿 bayes_reg_mods_predictions.csv.

It pulls our model predictions using a custom function bayes_mod_predictions.

# setwd(mod_data_wd) bayes_reg_mods_predictions <- bayes_mod_predictions(models
# = bayes_reg_mods, original_data = slope_tidy)
# write.csv(bayes_reg_mods_predictions, 'bayes_reg_mods_predictions.csv',
# row.names = FALSE)

Reading in the predicted data

setwd(mod_data_wd)
bayes_reg_mods_predictions <- read.csv("bayes_reg_mods_predictions.csv")

We are going to combined this with our best fitting model df, so we know how they ranks for LOO.

bayes_reg_mods_predictions <- left_join(bayes_reg_mods_predictions, best_fit_bayes_reg,
    by = c("id", "model_type"))

10.3.4 📈 Results

10.3.4.1 Model selection summary

The best fitting models was most often a 2nd-order polynomial (n = 25, 43%). This could suggest the presence of a critical oxygen threshold (Pcrit) where the relationship between O2 and MO2 changes. To confirm their is a Pcrit, we need to validated the shape of the polynomials and in should use a more specific model to test the Pcrit value. In any case, This type of model is indicative of oxyregulator. The same is also true for those with a 3rd-order polynomial (n = 12, 21%).

The next most common is 1st-order polynomial (n = 18, 31%). In the case of the 1st-order polynomials, it suggest the presences of linear relationship between o2 and MO2, which is indicative of oxyconformer. However, to be true evidence of a oxyconformer this relationship should be positive (i.e. as O2 falls MO2 also falls). 13 of the 18 individuals best modelled with a linear function had positive estimates with credible intervals that did not overlap with zero (Table S3).

Lastly, 0th-order was the least common (n = 3, 5%), it suggests that MO2 does not show a statistically significant dependence on the O2. In other words, the metabolic rate does not adjust based on oxygen availability, and there is no clear critical oxygen threshold (Pcrit) where the relationship changes. This is indicative of a oxyregulator.

best_mod <- best_fit_bayes_reg %>%
    dplyr::filter(elpd_loo_rank == 1)

total_fish <- nrow(best_mod)

mod_summary_table <- best_mod %>%
    dplyr::group_by(model_type) %>%
    dplyr::reframe(n = length(id), percent = round((n/total_fish) * 100, 2)) %>%
    dplyr::mutate(best_model_name = case_when(model_type == "lm_0" ~ "0th-order polynomial",
        model_type == "lm_1" ~ "1st-order polynomial", model_type == "lm_2" ~ "2nd-order polynomial",
        model_type == "lm_3" ~ "3rd-order polynomial", TRUE ~ "ERROR"))


table_bwm <- mod_summary_table %>%
    dplyr::select(best_model_name, everything(), -model_type) %>%
    gt() %>%
    cols_align(align = "center", columns = everything())

table_bwm
best_model_name n percent
0th-order polynomial 3 5.17
1st-order polynomial 18 31.03
2nd-order polynomial 25 43.10
3rd-order polynomial 12 20.69

Summary of fish best model with a linear function.

10.3.5 Table S3

Table S3: Ten fish that were best modelled with a linear function, showing r2, and estimate (). Only two fish had positive estimates with credible intervals that did not overlap with zero, which are highlighted as conforming in the table. Thus in total, 13 of 58 fish showed sufficient evidence to be conforming.

table_lm_1 <- best_mod %>%
    dplyr::filter(model_type == "lm_1") %>%
    dplyr::mutate(r_sq_ci = paste0(round(r2, 3), " (", round(r2_q2.5, 3), "–",
        round(r2_q97.5, 3), ")"), est_ci = paste0(round(estimate_DO, 6), " (", round(conf.low_DO,
        6), "–", round(conf.high_DO, 6), ")"), conformer = if_else(conf.low_DO >
        0, "Conforming", "Not conforming")) %>%
    dplyr::select(id, r_sq_ci, est_ci, conformer) %>%
    dplyr::arrange(conformer) %>%
    dplyr::ungroup()


table_lm_1 %>%
    gt() %>%
    cols_align(align = "center", columns = everything()) %>%
    cols_label(id = "Fish ID", r_sq_ci = "r2 (CI)", est_ci = "Estimate (CI)", conformer = "Evidence of oxyconforming")
Fish ID r2 (CI) Estimate (CI) Evidence of oxyconforming
a_0_24nov_1 0.187 (0.022–0.373) 0.000998 (0.000331–0.001701) Conforming
a_0_24nov_3 0.254 (0.058–0.428) 0.001358 (0.000601–0.002094) Conforming
a_0_25nov_1 0.537 (0.32–0.658) 0.000881 (0.000581–0.001173) Conforming
a_0_25nov_4 0.253 (0.049–0.441) 0.000847 (0.000346–0.001356) Conforming
a_9_22nov_1 0.198 (0.009–0.399) 0.00062 (0.000124–0.001096) Conforming
b_0_25nov_2 0.451 (0.223–0.595) 0.000856 (0.000523–0.001177) Conforming
b_0_27nov_3 0.254 (0.024–0.462) 0.000362 (0.000104–0.000614) Conforming
b_9_22nov_1 0.421 (0.23–0.556) 0.000855 (0.00056–0.001146) Conforming
c_9_24nov_4 0.294 (0.102–0.459) 0.000422 (0.000227–0.000623) Conforming
c_9_25nov_4 0.535 (0.335–0.655) 0.000455 (0.00031–0.000601) Conforming
c_9_27nov_2 0.186 (0.003–0.411) 0.00062 (7e-06–0.001206) Conforming
d_0_22nov_3 0.141 (0.002–0.33) 0.000356 (2.6e-05–0.000677) Conforming
d_9_25nov_3 0.294 (0.053–0.489) 0.001461 (0.000569–0.002336) Conforming
a_9_21nov_1 0.131 (0.001–0.338) 0.000314 (-4.3e-05–0.000668) Not conforming
b_0_24nov_4 0.101 (0.001–0.271) 0.000601 (-4e-05–0.001244) Not conforming
b_0_27nov_2 0.136 (0.002–0.348) 0.000235 (-1.5e-05–0.000492) Not conforming
c_0_22nov_3 0.109 (0–0.323) 0.000762 (-0.000341–0.001872) Not conforming
d_9_25nov_2 0.124 (0.001–0.33) 0.000386 (-5.9e-05–0.000828) Not conforming

10.3.5.1 Ploting all models

Data set with all slopes and which model was best

best_fit <- left_join(slope_tidy, best_mod, by = "id")

Now we are plotting each of the regressions. First making a directory to save the figures

Plot all regression, and highlighting the model that has the best fit, based on AIC values

# Create a list to store the plots
plots <- list()
model_preds_list <- list()

for (id_i in ids) {
  
  # Filter data for the current ID
  df_i <- bayes_reg_mods_predictions %>%
    dplyr::filter(id == id_i) %>% 
    dplyr::mutate(line_size = if_else(elpd_loo_rank == 1, 2, 1),
           alpha_value = if_else(elpd_loo_rank == 1, 1, 0.4))
  
  x_min <- df_i %>%
    dplyr::reframe(min = min(DO), na.rm = TRUE) %>% 
    dplyr::pull(min)
  
  y_max <- df_i %>%
    dplyr::reframe(max = max(MO2_g), na.rm = TRUE) %>% 
    dplyr::pull(max)
  
  best_weighted_model_i <- best_fit_bayes_reg %>% 
    dplyr::filter(id == id_i & elpd_loo_rank == 1)
  
  poly_i_name <- best_weighted_model_i %>%
    dplyr::mutate(name = case_when(
      model_type == "lm_0" ~ "0th-order",
      model_type == "lm_1" ~ "1st-order",
      model_type == "lm_2" ~ "2nd-order",
      model_type == "lm_3" ~ "3rd-order",
      TRUE ~ "ERROR"
    )) %>% 
    dplyr::pull(name)
  
  r_i <- best_weighted_model_i %>% 
    dplyr::mutate(r_sq_ci = paste0(round(r2, 3), " (", 
                                    round(r2_q2.5, 3), "–", 
                                    round(r2_q97.5, 3), ")")) %>% 
    dplyr::pull(r_sq_ci)

  # Create the plot
  plot_i <- ggplot() +
    geom_ribbon(data = df_i,
                aes(x = DO, y = predicted, ymin = pred_lower, ymax = pred_upper, fill = model_type),
                alpha = 0.1) +
    geom_line(data = df_i, 
              aes(x = DO, y = predicted, colour = model_type, size = line_size, alpha = alpha_value)) +
    geom_point(data = df_i %>% dplyr::filter(elpd_loo_rank == 1), aes(x = DO, y = MO2_g), alpha = 0.6, colour = "black", size = 2) +
    scale_colour_manual(values = c("red", "blue", "green", "purple"), 
                        labels = c("0th Order", "1st Order", "2nd Order", "3rd Order")) +
    scale_size_identity() +  # Use the size values directly
    scale_alpha_identity(guide = "none") +  # Remove the alpha legend 
    annotate("text", x = x_min, 
             y = y_max, 
             label = paste0("Best fit: ",poly_i_name, "\n", "r2 = ", r_i), 
             hjust = 0, vjust = 1, size = 4) +
    labs(
      title = paste(id_i),
      x = "Dissolved oxygen percentage (DO)",
      y = "MO2 (o2 mg/g/h)",
      colour = "Model") +
    theme_classic() +
    theme(legend.position = "none")
  
  # Store the plot
  plots[[id_i]] <- plot_i
  
  print(plot_i)
}

pdf(file = paste0(incremental_reg_bayes_wd, "./combined_reg_plots.pdf"), width = 8, height = 6)

for (id_i in ids) {
  print(plots[[id_i]])  # Print each plot to the PDF
}

dev.off()  # Close the PDF device
## png 
##   2

Getting a plot for each best fit regression, and overlaying a global model for that model type.

Here we are grouping fish by best fitting model and getting an average trend. I have hashed the code so you don’t need to re-run the models

⏭️ Skip this step if you have downloaded the global models ‘.rds’ in the ‘bayes-regs-global’ folder

# setwd(output_mods_bayes_global_wd) seed_number = 143019 # Set up parallel
# backend plan(multisession, workers = 4) # Adjust workers based on system
# resources # Define model formulas and data model_formulas <- list( lm_0 =
# bf(MO2_g ~ 1, family = gaussian()), lm_1 = bf(MO2_g ~ DO, family =
# gaussian()), lm_2 = bf(MO2_g ~ poly(DO, 2), family = gaussian()), lm_3 =
# bf(MO2_g ~ poly(DO, 3), family = gaussian()) ) model_data <- list( lm_0 =
# best_fit %>% filter(model_type == 'lm_0'), lm_1 = best_fit %>%
# filter(model_type == 'lm_1'), lm_2 = best_fit %>% filter(model_type ==
# 'lm_2'), lm_3 = best_fit %>% filter(model_type == 'lm_3') ) # Run models in
# parallel with future_map() models <- future_map(names(model_formulas), ~{
# brm( formula = model_formulas[[.x]], data = model_data[[.x]], cores = 4, seed
# = seed_number, save_pars = save_pars(all = TRUE), sample_prior = FALSE,
# silent = TRUE, file = paste0(.x, '_global') ) }) # Assign model names to
# results names(models) <- names(model_formulas) # Stop parallel plan after
# execution plan(sequential)

Loading these models

bayes_reg_mods_global <- load_rds(model_dw = output_mods_bayes_global_wd)

Pulling predictions

global_models_preds <- list()

for (mode_name in names(bayes_reg_mods_global)) {

    mod_i <- bayes_reg_mods_global[[mode_name]]
    mode_type = str_remove(mode_name, "_global")

    model_predictions_i <- as.data.frame(fitted(mod_i, summary = TRUE)) %>%
        dplyr::mutate(model = mode_name, model_type = mode_type) %>%
        dplyr::rename(pred_lower = Q2.5, pred_upper = Q97.5, predicted = Estimate,
            pred_error = Est.Error) %>%
        dplyr::select(model, everything())

    original_data_i <- best_fit %>%
        filter(model_type == mode_type) %>%
        dplyr::select(-model_type)

    model_predictions_original_i <- cbind(model_predictions_i, original_data_i)

    global_models_preds[[mode_name]] <- model_predictions_original_i
}

global_models_pred_df <- bind_rows(global_models_preds)

10.3.5.2 Figure S10

Best fit regressions with global models based on that polynomial order.

Figure S10 is also called Figure 2 in the manuscript.

global_models_pred_df <- global_models_pred_df %>%
  dplyr::mutate(model_name = case_when(
      model_type == "lm_0" ~ "0th-order polynomial",
      model_type == "lm_1" ~ "1st-order polynomial",
      model_type == "lm_2" ~ "2nd-order polynomial",
      model_type == "lm_3" ~ "3rd-order polynomial",
      TRUE ~ "ERROR"
    ))

annotation_data <- mod_summary_table %>%
  dplyr::select(model_type, best_model_name, n)

bayes_reg_mods_predictions_best <- bayes_reg_mods_predictions %>% 
  dplyr::filter(elpd_loo_rank == 1) 

fig_1 <- ggplot() +
  geom_line(data = bayes_reg_mods_predictions_best,
            aes(x = DO, y = predicted, color = id), size = 1) +
  # geom_smooth(data = best_fit,
  #           aes(x = DO, y = MO2_g, color = id), size = 1, alpha = 1, se = FALSE) +
  geom_ribbon(data = global_models_pred_df,
              aes(x = DO, ymin = pred_lower
, ymax = pred_upper, group = model),
              fill = "#FC6C85", alpha = 0.2) +  # Shaded confidence intervals
  geom_line(data = global_models_pred_df,
            aes(x = DO, y = predicted), size = 1.5, color = "#FF007F") +
  facet_wrap(~model_type) +
  scale_color_grey(start = 0.1, end = 0.9) +
  labs(
      title = paste("Best fit regressions grouped by polynomial order"),
      x = "Dissolved oxygen percentage (DO)",
      y = "MO2 (O2 mg/g/h)") +
  theme_classic() +
  theme(legend.position = "none") +
  geom_text(data = annotation_data,
            aes(x = -Inf, y = Inf, label = paste0("italic(n) == ", n)),
            hjust = -0.1, vjust = 1.2, inherit.aes = FALSE, parse = TRUE)
fig_1

Figure 2: Best fit models from incremental regression analyses (0th to 3 rd polynomial regressions) grouped by polynomial order, the pink regression represents the average for all fish fit with that polynomial order.

10.4 Pcrit model

For those fish that were best modelled with a 2nd or 3rd-order polynomial (n = 37) we will check to see if a Pcrit is present. We are filtering the data for only those fish.

higher_order <- c("lm_2", "lm_3")

check_pcrit_ids <- best_mod %>%
    dplyr::filter(model_type %in% higher_order) %>%
    dplyr::distinct(id) %>%
    pull(id)

check_pcrit_df <- slope_tidy %>%
    dplyr::filter(id %in% check_pcrit_ids)

We will calculate Pcrit using Chabot method and function calcO2crit.

This function uses the fifth percentile of the MO2 values observed at dissolved oxygen levels ≥ 80% air saturation as the criterion to assess low MO2 values. The algorithm then identifies all the MO2 measurements greater than this minimally acceptable MO2 value. Within this sub-set, it identifies the ̇ MO2 measurement made at the lowest DO and thereafter considers this DO as candidate for breakpoint (named pivotDO in the script). A regression is then calculated using observations at DO levels < pivotDO, and a first estimate of O2crit is calculated as the intersection of this regression line with the horizontal line representing SMR. The script then goes through validation steps to ensure that the slope of the regression is not so low that the line, projected to normoxic DO levels, passes below any MO2 values observed in normoxia. It also ensures that the intercept is not greater than zero. Corrective measures are taken if such problems are encountered.

lowestMO2 default is the quantile(MO2[DO >= 80], p=0.05). It is used to segment the data and locate the pivotDO.

ids <- check_pcrit_df %>%
    dplyr::distinct(id) %>%
    dplyr::pull()

pcrit_model_df_list <- list()
pcrit_models <- list()

for (id_i in ids) {

    df_i <- check_pcrit_df %>%
        dplyr::filter(id == id_i)

    o2crit <- calcO2crit(Data = df_i, SMR = df_i$SMR[1], lowestMO2 = NA, gapLimit = 4,
        max.nb.MO2.for.reg = 7)

    vaule <- o2crit$o2crit
    lowestMO2 = quantile(df_i$MO2[df_i$DO >= 80], p = 0.05)
    SMR <- o2crit$SMR
    nb_mo2_conforming <- o2crit$Nb_MO2_conforming
    r2 <- o2crit$r2
    method <- o2crit$Method
    p <- o2crit$P[1]

    pcrit_model_df <- tibble(id = id_i, pcrit_vaule = vaule, pcrit_smr = SMR, pcrit_lowestMO2 = lowestMO2,
        pcrit_nb_mo2_conforming = nb_mo2_conforming, pcrit_r2 = r2, pcrit_method = method,
        pcrit_p = p)

    pcrit_model_df_list[[id_i]] <- pcrit_model_df

    pcrit_models[[id_i]] <- o2crit

}

pcrit_model_df <- bind_rows(pcrit_model_df_list)

10.4.1 Ploting Pcrit

Here’s the plots for the Pcrit estimates. This includes all fish with best fit regression that are 2 or 3 order (n = 37).

## png 
##   2

Printing in htlm document

ids <- check_pcrit_df %>%
    dplyr::distinct(id) %>%
    dplyr::pull()

for (id_i in ids) {

    comment <- check_pcrit_df %>%
        dplyr::filter(id == id_i) %>%
        dplyr::slice(1) %>%
        dplyr::mutate(comment = if_else(is.na(comments), "", paste0("#", comments))) %>%
        pull(comment)

    r2 <- pcrit_model_df %>%
        dplyr::filter(id == id_i) %>%
        dplyr::mutate(pcrit_r2 = round(pcrit_r2, 3)) %>%
        dplyr::pull(pcrit_r2)

    conforming <- pcrit_model_df %>%
        dplyr::filter(id == id_i) %>%
        dplyr::mutate(pcrit_nb_mo2_conforming = round(pcrit_nb_mo2_conforming, 3)) %>%
        dplyr::pull(pcrit_nb_mo2_conforming)

    P <- pcrit_model_df %>%
        dplyr::filter(id == id_i) %>%
        dplyr::mutate(pcrit_p = round(pcrit_p, 3)) %>%
        dplyr::pull(pcrit_p)

    SMR <- pcrit_model_df %>%
        dplyr::filter(id == id_i) %>%
        dplyr::mutate(pcrit_smr = round(pcrit_smr, 3)) %>%
        dplyr::pull(pcrit_smr)

    lowestMO2 <- pcrit_model_df %>%
        dplyr::filter(id == id_i) %>%
        dplyr::mutate(pcrit_lowestMO2 = round(pcrit_lowestMO2, 3)) %>%
        dplyr::pull(pcrit_lowestMO2)

    # Generate and render the plot
    plotO2crit(o2critobj = pcrit_models[[id_i]])

    # Add a title
    mtext(text = paste0(id_i, " ", comment), side = 3, line = 2, adj = 0, col = "blue",
        font = 2, cex = 1.2)

    mtext(text = paste0("R2 = ", r2, "; p = ", P, "; CP < SMR = ", conforming, "; SMR = ",
        SMR, "; lowestMO2 = ", lowestMO2), side = 3, line = 1, adj = 0, col = "blue",
        font = 0.5, cex = 0.8)
}

10.4.2 🚧 Visual inspection 🚧

🚧 Need to re-do visual inspections with new data 🚧

We conducted visual inspection of all 37 fish to determine if Pcrits were present. We have loaded this as a data frame 💿 pcrit_check

For this visual inspection, we grouped fish into “no”, “unclear” or “yes”, based on the certainty at which we observed a Pcrit. We concluded that 12 fish had a Pcrit responses, while 6 showed some evidence of a Pcrit but not as clearly (i.e. unclear), and the remaining 20 did not show any evidence of a Pcrit response.

pcrit_check %>%
    dplyr::group_by(pcrit) %>%
    dplyr::reframe(n = length(id)) %>%
    gt()
pcrit n
no 20
unclear 6
yes 12

Let’s make a list of those that we did visually identify as a Pcrit

pcrit_visual_yes <- pcrit_check %>%
    dplyr::filter(pcrit == "yes") %>%
    dplyr::pull(id)

10.4.3 Pcrit numerical rules

We also used a numerical rule to asses if a Pcrit was observed.

We filtered for only cases were at the lowest O₂ value three consecutive MO₂ measures full below our SMR and fifth percentile of the MO2 values observed at dissolved O2 levels > 80%. In the model output these are called nb_mo2_conforming points.

pcrit_numerical_yes <- pcrit_model_df %>%
    dplyr::filter(pcrit_nb_mo2_conforming > 2) %>%
    pull(id)

paste0("Based on this rule there are ", length(pcrit_numerical_yes), " fish with possible Pcrits.")
## [1] "Based on this rule there are 21 fish with possible Pcrits."

10.4.4 🚧 📈 Pcrits numbers 🚧

🚧 Need to re-do visual inspections with new data 🚧

We will now check to see which fish have Pcrits based on the numerical and visual inspections.

shared_elements <- intersect(pcrit_visual_yes, pcrit_numerical_yes)

unique_to_visual <- setdiff(pcrit_visual_yes, pcrit_numerical_yes)

unique_to_numerical <- setdiff(pcrit_numerical_yes, pcrit_visual_yes)

list(Shared = shared_elements, Unique_to_visual = unique_to_visual, Unique_to_numerical = unique_to_numerical)
## $Shared
##  [1] "a_0_26nov_4" "a_9_21nov_3" "b_0_24nov_1" "b_0_24nov_2" "b_0_25nov_1"
##  [6] "b_0_25nov_3" "b_0_26nov_1" "b_9_21nov_1" "b_9_21nov_2" "b_9_21nov_3"
## [11] "d_0_21nov_3"
## 
## $Unique_to_visual
## [1] "d_0_21nov_2"
## 
## $Unique_to_numerical
##  [1] "a_0_24nov_4" "b_0_25nov_4" "b_0_26nov_3" "b_9_21nov_4" "b_9_22nov_3"
##  [6] "c_0_21nov_2" "c_9_24nov_2" "c_9_27nov_4" "d_0_22nov_2" "d_9_24nov_3"

There are 10 fish that both numerically and visually were determined to have a Pcrit. Two fish were detriment visually to have a Pcrit, while they did not meet the numerical criteria. Three fish meet the numerical criteria, but did not pass the visual inspection.

These are the 10 fish that meet numerical criteria and passed visual inspection.

for (id_i in shared_elements) {

    comment <- check_pcrit_df %>%
        dplyr::filter(id == id_i) %>%
        dplyr::slice(1) %>%
        dplyr::mutate(comment = if_else(is.na(comments), "", paste0("#", comments))) %>%
        pull(comment)

    r2 <- pcrit_model_df %>%
        dplyr::filter(id == id_i) %>%
        dplyr::mutate(pcrit_r2 = round(pcrit_r2, 3)) %>%
        dplyr::pull(pcrit_r2)

    conforming <- pcrit_model_df %>%
        dplyr::filter(id == id_i) %>%
        dplyr::mutate(pcrit_nb_mo2_conforming = round(pcrit_nb_mo2_conforming, 3)) %>%
        dplyr::pull(pcrit_nb_mo2_conforming)

    P <- pcrit_model_df %>%
        dplyr::filter(id == id_i) %>%
        dplyr::mutate(pcrit_p = round(pcrit_p, 3)) %>%
        dplyr::pull(pcrit_p)

    SMR <- pcrit_model_df %>%
        dplyr::filter(id == id_i) %>%
        dplyr::mutate(pcrit_smr = round(pcrit_smr, 3)) %>%
        dplyr::pull(pcrit_smr)

    lowestMO2 <- pcrit_model_df %>%
        dplyr::filter(id == id_i) %>%
        dplyr::mutate(pcrit_lowestMO2 = round(pcrit_lowestMO2, 3)) %>%
        dplyr::pull(pcrit_lowestMO2)

    # Generate and render the plot
    plotO2crit(o2critobj = pcrit_models[[id_i]])

    # Add a title
    mtext(text = paste0(id_i, " ", comment), side = 3, line = 2, adj = 0, col = "blue",
        font = 2, cex = 1.2)

    mtext(text = paste0("R2 = ", r2, "; p = ", P, "; CP < SMR = ", conforming, "; SMR = ",
        SMR, "; lowestMO2 = ", lowestMO2), side = 3, line = 1, adj = 0, col = "blue",
        font = 1, cex = 0.8)
}

These are the 2 fish that did not meet numerical criteria but passed visual inspection

for (id_i in unique_to_visual) {

    comment <- check_pcrit_df %>%
        dplyr::filter(id == id_i) %>%
        dplyr::slice(1) %>%
        dplyr::mutate(comment = if_else(is.na(comments), "", paste0("#", comments))) %>%
        pull(comment)

    r2 <- pcrit_model_df %>%
        dplyr::filter(id == id_i) %>%
        dplyr::mutate(pcrit_r2 = round(pcrit_r2, 3)) %>%
        dplyr::pull(pcrit_r2)

    conforming <- pcrit_model_df %>%
        dplyr::filter(id == id_i) %>%
        dplyr::mutate(pcrit_nb_mo2_conforming = round(pcrit_nb_mo2_conforming, 3)) %>%
        dplyr::pull(pcrit_nb_mo2_conforming)

    P <- pcrit_model_df %>%
        dplyr::filter(id == id_i) %>%
        dplyr::mutate(pcrit_p = round(pcrit_p, 3)) %>%
        dplyr::pull(pcrit_p)

    SMR <- pcrit_model_df %>%
        dplyr::filter(id == id_i) %>%
        dplyr::mutate(pcrit_smr = round(pcrit_smr, 3)) %>%
        dplyr::pull(pcrit_smr)

    lowestMO2 <- pcrit_model_df %>%
        dplyr::filter(id == id_i) %>%
        dplyr::mutate(pcrit_lowestMO2 = round(pcrit_lowestMO2, 3)) %>%
        dplyr::pull(pcrit_lowestMO2)

    # Generate and render the plot
    plotO2crit(o2critobj = pcrit_models[[id_i]])

    # Add a title
    mtext(text = paste0(id_i, " ", comment), side = 3, line = 2, adj = 0, col = "blue",
        font = 2, cex = 1.2)

    mtext(text = paste0("R2 = ", r2, "; p = ", P, "; CP < SMR = ", conforming, "; SMR = ",
        SMR, "; lowestMO2 = ", lowestMO2), side = 3, line = 1, adj = 0, col = "blue",
        font = 1, cex = 0.8)
}

These are the 3 fish that did meet numerical criteria but did not pass visual inspection

for (id_i in unique_to_numerical) {

    comment <- check_pcrit_df %>%
        dplyr::filter(id == id_i) %>%
        dplyr::slice(1) %>%
        dplyr::mutate(comment = if_else(is.na(comments), "", paste0("#", comments))) %>%
        pull(comment)

    r2 <- pcrit_model_df %>%
        dplyr::filter(id == id_i) %>%
        dplyr::mutate(pcrit_r2 = round(pcrit_r2, 3)) %>%
        dplyr::pull(pcrit_r2)

    conforming <- pcrit_model_df %>%
        dplyr::filter(id == id_i) %>%
        dplyr::mutate(pcrit_nb_mo2_conforming = round(pcrit_nb_mo2_conforming, 3)) %>%
        dplyr::pull(pcrit_nb_mo2_conforming)

    P <- pcrit_model_df %>%
        dplyr::filter(id == id_i) %>%
        dplyr::mutate(pcrit_p = round(pcrit_p, 3)) %>%
        dplyr::pull(pcrit_p)

    SMR <- pcrit_model_df %>%
        dplyr::filter(id == id_i) %>%
        dplyr::mutate(pcrit_smr = round(pcrit_smr, 3)) %>%
        dplyr::pull(pcrit_smr)

    lowestMO2 <- pcrit_model_df %>%
        dplyr::filter(id == id_i) %>%
        dplyr::mutate(pcrit_lowestMO2 = round(pcrit_lowestMO2, 3)) %>%
        dplyr::pull(pcrit_lowestMO2)

    # Generate and render the plot
    plotO2crit(o2critobj = pcrit_models[[id_i]])

    # Add a title
    mtext(text = paste0(id_i, " ", comment), side = 3, line = 2, adj = 0, col = "blue",
        font = 2, cex = 1.2)

    mtext(text = paste0("R2 = ", r2, "; p = ", P, "; CP < SMR = ", conforming, "; SMR = ",
        SMR, "; lowestMO2 = ", lowestMO2), side = 3, line = 1, adj = 0, col = "blue",
        font = 1, cex = 0.8)
}

Based on the 10 fish that passed both the numerical criteria and visually checks the Pcrit response.

n_pcrit <- length(shared_elements)

have_pcirt <- pcrit_model_df %>%
    dplyr::filter(id %in% shared_elements)

mean_pcrit <- have_pcirt %>%
    dplyr::reframe(mean = mean(pcrit_vaule)) %>%
    pull(mean) %>%
    round(., 2)

min_pcrit <- have_pcirt %>%
    dplyr::reframe(min = min(pcrit_vaule)) %>%
    pull(min) %>%
    round(., 2)

max_pcrit <- have_pcirt %>%
    dplyr::reframe(max = max(pcrit_vaule)) %>%
    pull(max) %>%
    round(., 2)

print(paste0("There are ", n_pcrit, " fish with identified Pcrits and the mean Pcrit is ",
    mean_pcrit, " (range: ", min_pcrit, "-", max_pcrit, ")", " however, the true range for Pcrit is likey much larger then reported here, as many fish did not exhibit a Pcrit like response with the trial"))
## [1] "There are 11 fish with identified Pcrits and the mean Pcrit is 32.78 (range: 21.3-57.4) however, the true range for Pcrit is likey much larger then reported here, as many fish did not exhibit a Pcrit like response with the trial"

The mean lowest dissolved oxygen (DO) value for trails where a Pcrit was not detected was 13.11% ± 3.30% (mean ± sd), the lowest recoded DO value for which a Pcrit was not detected was 8.82%. Thus the true Pcrit range of Galaxias maculatus is 39.9% to < 8.83%

slope_tidy %>%
    dplyr::filter(!(id %in% shared_elements)) %>%
    dplyr::group_by(id) %>%
    dplyr::arrange(DO) %>%
    dplyr::slice(1) %>%
    dplyr::ungroup() %>%
    dplyr::reframe(mean_lowest_o2 = mean(DO), sd_lowest_o2 = sd(DO), min_o2 = min(DO))
## # A tibble: 1 × 3
##   mean_lowest_o2 sd_lowest_o2 min_o2
##            <dbl>        <dbl>  <dbl>
## 1           13.1         3.33   8.82

For the fish that did have Pcrits, proportionally more fish were from the zero salinity treatment (20% vs 14%), and the fish were close to the mean weight.

mean_weight_all <- slope_tidy %>%
    dplyr::filter(!(id %in% shared_elements)) %>%
    dplyr::group_by(id) %>%
    dplyr::slice(1) %>%
    dplyr::ungroup() %>%
    dplyr::reframe(mean_weight_all = mean(mass, na.rm = TRUE)) %>%
    dplyr::pull(mean_weight_all)

total_salinity <- slope_tidy %>%
    dplyr::group_by(id) %>%
    dplyr::slice(1) %>%
    dplyr::ungroup() %>%
    dplyr::group_by(salinity_group) %>%
    dplyr::reframe(n = length(id))

n_salinity_0 <- total_salinity %>%
    dplyr::filter(salinity_group == "0") %>%
    dplyr::pull(n)

n_salinity_9 <- total_salinity %>%
    dplyr::filter(salinity_group == "9") %>%
    dplyr::pull(n)

slope_tidy %>%
    dplyr::filter(id %in% shared_elements) %>%
    dplyr::group_by(id) %>%
    dplyr::slice(1) %>%
    dplyr::ungroup() %>%
    dplyr::reframe(n_pcrit_salinity_9 = sum(salinity_group == "9", na.rm = TRUE),
        n_pcrit_salinity_0 = sum(salinity_group == "0", na.rm = TRUE), prop_salinity_9 = n_pcrit_salinity_9/n_salinity_9,
        prop_salinity_0 = n_pcrit_salinity_0/n_salinity_0, rel_mean_weight = mean(mass,
            na.rm = TRUE)/mean_weight_all) %>%
    gt()
n_pcrit_salinity_9 n_pcrit_salinity_0 prop_salinity_9 prop_salinity_0 rel_mean_weight
4 7 0.1428571 0.2333333 0.903909

11 📑 References

[1] Claireaux, G. and Chabot, D. (2016) Responses by fishes to environmental hypoxia: integration through Fry’s concept of aerobic metabolic scope. Journal of Fish Biology https://doi.org/10.1111/jfb.12833

[2] Urbina MA, Glover CN, and Forster ME, (2012) A novel oxyconforming response in the freshwater fish Galaxias maculatus. Comparative Biochemistry and Physiology Part A: Molecular & Integrative Physiology. https://doi.org/10.1016/j.cbpa.2011.11.011

[3] Pick JL, Nakagawa S, and Noble DWA (2018) Reproducible, flexible and high-throughput data extraction from primary literature: The metaDigitise r package. https://doi.org/10.1111/2041-210X.13118

12 💻 Session information

Here is a detailed list of the session information

sessioninfo::session_info()
## ─ Session info ───────────────────────────────────────────────────────────────
##  setting  value
##  version  R version 4.2.3 (2023-03-15 ucrt)
##  os       Windows 10 x64 (build 19044)
##  system   x86_64, mingw32
##  ui       RTerm
##  language (EN)
##  collate  English_Australia.utf8
##  ctype    English_Australia.utf8
##  tz       Australia/Sydney
##  date     2025-02-18
##  pandoc   3.1.1 @ C:/Program Files/RStudio/resources/app/bin/quarto/bin/tools/ (via rmarkdown)
## 
## ─ Packages ───────────────────────────────────────────────────────────────────
##  package           * version    date (UTC) lib source
##  abind               1.4-5      2016-07-21 [1] CRAN (R 4.2.0)
##  arrayhelpers        1.1-0      2020-02-04 [1] CRAN (R 4.2.3)
##  backports           1.4.1      2021-12-13 [1] CRAN (R 4.2.0)
##  bayesplot         * 1.11.1     2024-02-15 [1] CRAN (R 4.2.3)
##  betareg           * 3.2-0      2024-07-07 [1] CRAN (R 4.2.3)
##  bitops              1.0-7      2021-04-24 [1] CRAN (R 4.2.0)
##  boot                1.3-28.1   2022-11-22 [2] CRAN (R 4.2.3)
##  bridgesampling      1.1-2      2021-04-16 [1] CRAN (R 4.2.3)
##  brms              * 2.21.0     2024-03-20 [1] CRAN (R 4.2.3)
##  Brobdingnag         1.2-9      2022-10-19 [1] CRAN (R 4.2.3)
##  broom             * 1.0.4      2023-03-11 [1] CRAN (R 4.2.3)
##  broom.helpers       1.15.0     2024-04-05 [1] CRAN (R 4.2.3)
##  broom.mixed       * 0.2.9.5    2024-04-01 [1] CRAN (R 4.2.3)
##  bslib               0.7.0      2024-03-29 [1] CRAN (R 4.2.3)
##  cachem              1.0.7      2023-02-24 [1] CRAN (R 4.2.3)
##  car               * 3.1-2      2023-03-30 [1] CRAN (R 4.2.3)
##  carData           * 3.0-5      2022-01-06 [1] CRAN (R 4.2.3)
##  caTools             1.18.2     2021-03-28 [1] CRAN (R 4.2.3)
##  cellranger          1.1.0      2016-07-27 [1] CRAN (R 4.2.3)
##  checkmate           2.3.1      2023-12-04 [1] CRAN (R 4.2.3)
##  cli                 3.6.1      2023-03-23 [1] CRAN (R 4.2.3)
##  cluster             2.1.4      2022-08-22 [2] CRAN (R 4.2.3)
##  coda                0.19-4.1   2024-01-31 [1] CRAN (R 4.2.3)
##  codetools           0.2-19     2023-02-01 [2] CRAN (R 4.2.3)
##  colorspace          2.1-0      2023-01-23 [1] CRAN (R 4.2.3)
##  corrplot          * 0.92       2021-11-18 [1] CRAN (R 4.2.3)
##  crosstalk           1.2.1      2023-11-23 [1] CRAN (R 4.2.3)
##  curl                5.2.1      2024-03-01 [1] CRAN (R 4.2.3)
##  dagitty           * 0.3-4      2023-12-07 [1] CRAN (R 4.2.3)
##  data.table        * 1.14.8     2023-02-17 [1] CRAN (R 4.2.3)
##  datawizard        * 0.12.0     2024-07-11 [1] CRAN (R 4.2.3)
##  DEoptimR            1.1-3      2023-10-07 [1] CRAN (R 4.2.3)
##  devtools          * 2.4.5      2022-10-11 [1] CRAN (R 4.2.3)
##  digest              0.6.31     2022-12-11 [1] CRAN (R 4.2.3)
##  distributional      0.4.0      2024-02-07 [1] CRAN (R 4.2.3)
##  doParallel          1.0.17     2022-02-07 [1] CRAN (R 4.2.3)
##  dplyr             * 1.1.1      2023-03-22 [1] CRAN (R 4.2.3)
##  DT                * 0.33       2024-04-04 [1] CRAN (R 4.2.3)
##  ellipsis            0.3.2      2021-04-29 [1] CRAN (R 4.2.3)
##  emmeans           * 1.10.3     2024-07-01 [1] CRAN (R 4.2.3)
##  estimability        1.5.1      2024-05-12 [1] CRAN (R 4.2.3)
##  evaluate            0.24.0     2024-06-10 [1] CRAN (R 4.2.3)
##  fansi               1.0.4      2023-01-22 [1] CRAN (R 4.2.3)
##  farver              2.1.1      2022-07-06 [1] CRAN (R 4.2.3)
##  fastmap             1.1.1      2023-02-24 [1] CRAN (R 4.2.3)
##  flexmix             2.3-19     2023-03-16 [1] CRAN (R 4.2.3)
##  fontawesome         0.5.2      2023-08-19 [1] CRAN (R 4.2.3)
##  forcats           * 1.0.0      2023-01-29 [1] CRAN (R 4.2.3)
##  foreach             1.5.2      2022-02-02 [1] CRAN (R 4.2.3)
##  formatR             1.14       2023-01-17 [1] CRAN (R 4.2.3)
##  Formula             1.2-5      2023-02-24 [1] CRAN (R 4.2.2)
##  fs                  1.6.4      2024-04-25 [1] CRAN (R 4.2.3)
##  furrr             * 0.3.1      2022-08-15 [1] CRAN (R 4.2.3)
##  future            * 1.33.2     2024-03-26 [1] CRAN (R 4.2.3)
##  generics            0.1.3      2022-07-05 [1] CRAN (R 4.2.3)
##  ggdag             * 0.2.12     2024-03-08 [1] CRAN (R 4.2.3)
##  ggdist              3.3.2      2024-03-05 [1] CRAN (R 4.2.3)
##  ggExtra           * 0.10.0     2022-03-23 [1] CRAN (R 4.2.3)
##  ggfortify         * 0.4.16     2023-03-20 [1] CRAN (R 4.2.3)
##  gghalves          * 0.1.4      2022-11-20 [1] CRAN (R 4.2.3)
##  ggplot2           * 3.5.1      2024-04-23 [1] CRAN (R 4.2.3)
##  ggridges          * 0.5.6      2024-01-23 [1] CRAN (R 4.2.3)
##  ggthemes          * 4.2.4      2021-01-20 [1] CRAN (R 4.2.3)
##  globals             0.16.3     2024-03-08 [1] CRAN (R 4.2.3)
##  glue                1.6.2      2022-02-24 [1] CRAN (R 4.2.3)
##  gridExtra         * 2.3        2017-09-09 [1] CRAN (R 4.2.3)
##  gsw                 1.2-0      2024-08-19 [1] CRAN (R 4.2.3)
##  gt                * 0.11.0     2024-07-09 [1] CRAN (R 4.2.3)
##  gtable              0.3.5      2024-04-22 [1] CRAN (R 4.2.3)
##  gtExtras          * 0.4.5      2022-11-28 [1] CRAN (R 4.2.3)
##  gtsummary         * 1.7.2      2023-07-15 [1] CRAN (R 4.2.3)
##  highr               0.11       2024-05-26 [1] CRAN (R 4.2.3)
##  hms               * 1.1.3      2023-03-21 [1] CRAN (R 4.2.3)
##  htmltools           0.5.8.1    2024-04-04 [1] CRAN (R 4.2.3)
##  htmlwidgets         1.6.2      2023-03-17 [1] CRAN (R 4.2.3)
##  httpuv              1.6.9      2023-02-14 [1] CRAN (R 4.2.3)
##  httr                1.4.7      2023-08-15 [1] CRAN (R 4.2.3)
##  igraph            * 1.6.0      2023-12-11 [1] CRAN (R 4.2.3)
##  inline              0.3.19     2021-05-31 [1] CRAN (R 4.2.3)
##  insight             0.20.1     2024-06-11 [1] CRAN (R 4.2.3)
##  iterators           1.0.14     2022-02-05 [1] CRAN (R 4.2.3)
##  janitor           * 2.2.0      2023-02-02 [1] CRAN (R 4.2.3)
##  jquerylib           0.1.4      2021-04-26 [1] CRAN (R 4.2.3)
##  jsonlite            1.8.4      2022-12-06 [1] CRAN (R 4.2.3)
##  knitr               1.42       2023-01-25 [1] CRAN (R 4.2.3)
##  labeling            0.4.3      2023-08-29 [1] CRAN (R 4.2.3)
##  later               1.3.0      2021-08-18 [1] CRAN (R 4.2.3)
##  lattice           * 0.20-45    2021-09-22 [2] CRAN (R 4.2.3)
##  lazyeval            0.2.2      2019-03-15 [1] CRAN (R 4.2.3)
##  lifecycle           1.0.4      2023-11-07 [1] CRAN (R 4.2.3)
##  listenv             0.9.1      2024-01-29 [1] CRAN (R 4.2.3)
##  lme4              * 1.1-35.5   2024-07-03 [1] CRAN (R 4.2.3)
##  lmerTest          * 3.1-3      2020-10-23 [1] CRAN (R 4.2.3)
##  lmtest              0.9-40     2022-03-21 [1] CRAN (R 4.2.3)
##  loo                 2.8.0      2024-07-03 [1] CRAN (R 4.2.3)
##  lubridate         * 1.9.2      2023-02-10 [1] CRAN (R 4.2.3)
##  magrittr            2.0.3      2022-03-30 [1] CRAN (R 4.2.3)
##  marelac             2.1.11     2023-09-25 [1] CRAN (R 4.2.3)
##  marginaleffects   * 0.21.0     2024-06-14 [1] CRAN (R 4.2.3)
##  MASS                7.3-58.2   2023-01-23 [2] CRAN (R 4.2.3)
##  Matrix            * 1.5-3      2022-11-11 [2] CRAN (R 4.2.3)
##  matrixStats         1.3.0      2024-04-11 [1] CRAN (R 4.2.3)
##  mclust            * 6.1.1      2024-04-29 [1] CRAN (R 4.2.3)
##  measurements        1.5.1      2023-04-28 [1] CRAN (R 4.2.3)
##  memoise             2.0.1      2021-11-26 [1] CRAN (R 4.2.3)
##  mgcv                1.9-1      2023-12-21 [1] CRAN (R 4.2.3)
##  mime                0.12       2021-09-28 [1] CRAN (R 4.2.0)
##  miniUI              0.1.1.1    2018-05-18 [1] CRAN (R 4.2.3)
##  minqa               1.2.7      2024-05-20 [1] CRAN (R 4.2.3)
##  modeltools          0.2-23     2020-03-05 [1] CRAN (R 4.2.0)
##  multcomp            1.4-25     2023-06-20 [1] CRAN (R 4.2.3)
##  munsell             0.5.1      2024-04-01 [1] CRAN (R 4.2.3)
##  mvtnorm             1.2-5      2024-05-21 [1] CRAN (R 4.2.3)
##  nlme                3.1-162    2023-01-31 [2] CRAN (R 4.2.3)
##  nloptr              2.1.1      2024-06-25 [1] CRAN (R 4.2.3)
##  nnet                7.3-18     2022-09-28 [2] CRAN (R 4.2.3)
##  numDeriv            2016.8-1.1 2019-06-06 [1] CRAN (R 4.2.0)
##  oce                 1.8-3      2024-08-17 [1] CRAN (R 4.2.3)
##  opdisDownsampling   1.0.1      2024-04-15 [1] CRAN (R 4.2.3)
##  paletteer           1.6.0      2024-01-21 [1] CRAN (R 4.2.3)
##  parallelly          1.37.1     2024-02-29 [1] CRAN (R 4.2.3)
##  pbmcapply           1.5.1      2022-04-28 [1] CRAN (R 4.2.0)
##  performance       * 0.12.0     2024-06-08 [1] CRAN (R 4.2.3)
##  permute           * 0.9-7      2022-01-27 [1] CRAN (R 4.2.3)
##  pillar              1.9.0      2023-03-22 [1] CRAN (R 4.2.3)
##  pkgbuild            1.4.4      2024-03-17 [1] CRAN (R 4.2.3)
##  pkgconfig           2.0.3      2019-09-22 [1] CRAN (R 4.2.3)
##  pkgload             1.3.2      2022-11-16 [1] CRAN (R 4.2.3)
##  plotly            * 4.10.1     2022-11-07 [1] CRAN (R 4.2.3)
##  plyr                1.8.8      2022-11-11 [1] CRAN (R 4.2.3)
##  posterior           1.6.0      2024-07-03 [1] CRAN (R 4.2.3)
##  pracma              2.4.4      2023-11-10 [1] CRAN (R 4.2.3)
##  profvis             0.3.8      2023-05-02 [1] CRAN (R 4.2.3)
##  promises            1.2.0.1    2021-02-11 [1] CRAN (R 4.2.3)
##  purrr             * 1.0.1      2023-01-10 [1] CRAN (R 4.2.3)
##  qqconf              1.3.2      2023-04-14 [1] CRAN (R 4.2.3)
##  qqplotr           * 0.0.6      2023-01-25 [1] CRAN (R 4.2.3)
##  QuickJSR            1.3.0      2024-07-08 [1] CRAN (R 4.2.3)
##  R6                  2.5.1      2021-08-19 [1] CRAN (R 4.2.3)
##  RColorBrewer      * 1.1-3      2022-04-03 [1] CRAN (R 4.2.0)
##  Rcpp              * 1.0.10     2023-01-22 [1] CRAN (R 4.2.3)
##  RcppParallel        5.1.8      2024-07-06 [1] CRAN (R 4.2.3)
##  readr             * 2.1.4      2023-02-10 [1] CRAN (R 4.2.3)
##  readxl            * 1.4.3      2023-07-06 [1] CRAN (R 4.2.3)
##  rematch2            2.1.2      2020-05-01 [1] CRAN (R 4.2.3)
##  remotes             2.5.0      2024-03-17 [1] CRAN (R 4.2.3)
##  reshape2            1.4.4      2020-04-09 [1] CRAN (R 4.2.3)
##  respirometry      * 2.0.0      2024-07-18 [1] CRAN (R 4.2.3)
##  rlang               1.1.0      2023-03-14 [1] CRAN (R 4.2.3)
##  rmarkdown           2.21       2023-03-26 [1] CRAN (R 4.2.3)
##  robustbase          0.99-4-1   2024-09-27 [1] CRAN (R 4.2.3)
##  rstan             * 2.32.6     2024-03-05 [1] CRAN (R 4.2.3)
##  rstantools          2.4.0      2024-01-31 [1] CRAN (R 4.2.3)
##  rstudioapi          0.14       2022-08-22 [1] CRAN (R 4.2.3)
##  sandwich            3.1-0      2023-12-11 [1] CRAN (R 4.2.3)
##  sass                0.4.9      2024-03-15 [1] CRAN (R 4.2.3)
##  scales              1.3.0      2023-11-28 [1] CRAN (R 4.2.3)
##  seacarb             3.3.3      2024-02-15 [1] CRAN (R 4.2.3)
##  sessioninfo       * 1.2.2      2021-12-06 [1] CRAN (R 4.2.3)
##  shape               1.4.6.1    2024-02-23 [1] CRAN (R 4.2.3)
##  shiny             * 1.8.1.1    2024-04-02 [1] CRAN (R 4.2.3)
##  shinybusy         * 0.3.3      2024-03-09 [1] CRAN (R 4.2.3)
##  shinycssloaders   * 1.0.0      2020-07-28 [1] CRAN (R 4.2.3)
##  snakecase           0.11.1     2023-08-27 [1] CRAN (R 4.2.3)
##  SolveSAPHE          2.1.0      2021-05-01 [1] CRAN (R 4.2.0)
##  SRS               * 0.2.3      2022-03-27 [1] CRAN (R 4.2.3)
##  StanHeaders       * 2.32.9     2024-05-29 [1] CRAN (R 4.2.3)
##  stringi             1.7.12     2023-01-11 [1] CRAN (R 4.2.2)
##  stringr           * 1.5.1      2023-11-14 [1] CRAN (R 4.2.3)
##  survival            3.5-3      2023-02-12 [2] CRAN (R 4.2.3)
##  svUnit              1.0.6      2021-04-19 [1] CRAN (R 4.2.3)
##  tensorA             0.36.2.1   2023-12-13 [1] CRAN (R 4.2.3)
##  TH.data             1.1-2      2023-04-17 [1] CRAN (R 4.2.3)
##  tibble            * 3.2.1      2023-03-20 [1] CRAN (R 4.2.3)
##  tidybayes         * 3.0.6      2023-08-12 [1] CRAN (R 4.2.3)
##  tidygraph           1.3.0      2023-12-18 [1] CRAN (R 4.2.3)
##  tidyr             * 1.3.0      2023-01-24 [1] CRAN (R 4.2.3)
##  tidyselect          1.2.1      2024-03-11 [1] CRAN (R 4.2.3)
##  tidyverse         * 2.0.0      2023-02-22 [1] CRAN (R 4.2.3)
##  timechange          0.2.0      2023-01-11 [1] CRAN (R 4.2.3)
##  twosamples          2.0.1      2023-06-23 [1] CRAN (R 4.2.3)
##  tzdb                0.3.0      2022-03-28 [1] CRAN (R 4.2.3)
##  urlchecker          1.0.1      2021-11-30 [1] CRAN (R 4.2.3)
##  usethis           * 2.2.3      2024-02-19 [1] CRAN (R 4.2.3)
##  utf8                1.2.3      2023-01-31 [1] CRAN (R 4.2.3)
##  V8                  4.4.2      2024-02-15 [1] CRAN (R 4.2.3)
##  vctrs               0.6.1      2023-03-22 [1] CRAN (R 4.2.3)
##  vegan             * 2.6-6.1    2024-05-21 [1] CRAN (R 4.2.3)
##  viridisLite         0.4.2      2023-05-02 [1] CRAN (R 4.2.3)
##  withr               3.0.0      2024-01-16 [1] CRAN (R 4.2.3)
##  xfun                0.38       2023-03-24 [1] CRAN (R 4.2.3)
##  xml2                1.3.3      2021-11-30 [1] CRAN (R 4.2.3)
##  xtable              1.8-4      2019-04-21 [1] CRAN (R 4.2.3)
##  yaml                2.3.7      2023-01-23 [1] CRAN (R 4.2.3)
##  zoo                 1.8-12     2023-04-13 [1] CRAN (R 4.2.3)
## 
##  [1] C:/R
##  [2] C:/Program Files/R/R-4.2.3/library
## 
## ──────────────────────────────────────────────────────────────────────────────
LS0tDQp0aXRsZTogImdtYWMtbGFiLWNoYXJ0Ig0KYXV0aG9yOiAiSmFrZSBNYXJ0aW4iDQpkYXRlOiAiYHIgZm9ybWF0KFN5cy50aW1lKCksICclZCAlQiAlWScpYCINCm91dHB1dDoNCiAgaHRtbF9kb2N1bWVudDoNCiAgICBjb2RlX2Rvd25sb2FkOiB0cnVlDQogICAgY29kZV9mb2xkaW5nOiBoaWRlDQogICAgZGVwdGg6IDQNCiAgICBudW1iZXJfc2VjdGlvbnM6IHllcw0KICAgIHRoZW1lOiAgY29zbW8NCiAgICB0b2M6IHllcw0KICAgIHRvY19mbG9hdDogeWVzDQogICAgdG9jX2RlcHRoOiA0DQogIHBkZl9kb2N1bWVudDoNCiAgICB0b2M6IHllcw0Ka25pdDogfA0KICAoZnVuY3Rpb24oaW5wdXQsIC4uLikgew0KICAgIHJtYXJrZG93bjo6cmVuZGVyKA0KICAgICAgaW5wdXQsDQogICAgICBvdXRwdXRfZmlsZSA9IHBhc3RlMCgNCiAgICAgICAnaW5kZXguaHRtbCcNCiAgICAgICksDQogICAgICBlbnZpciA9IGdsb2JhbGVudigpDQogICAgKQ0KICB9KQ0KLS0tDQoNCjwhLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLT4NCiMg8J+TlSBSRUFEIE1FICAgDQo8IS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0+DQogICAgIA0KICAgICANCioq8J+apyBVTlBVQkxJU0hFRCBBTkQgT05HT0lORyDwn5qnICoqICAgDQpUaGUgZGF0YSBpbiB0aGlzIHNjcmlwdCBpcyBjdXJyZW50bHkgKip1bnB1Ymxpc2hlZCoqLCBhbmQgdGhlICoqYW5hbHlzaXMgaXMgb24gZ29pbmcqKi4gVGhlcmUgbWF5IGJlIHBhcnQgb2YgdGhpcyBzY3JpcHQgdGhhdCBhcmUgY3VycmVudGx5IGluY29tcGxldGUuIFRoaXMgc2NyaXB0IGhhcyBiZWVuIG1hZGUgcHVibGljbHkgYXZhaWxhYmxlIGZvciBlYXN5IG9mIHNoYXJpbmcgd2l0aCBjb2xsYWJvcmF0b3JzIGFuZCBjb2xsZWFndWVzIHdobyBhcmUgaW50ZXJlc3RlZCBpbiB0aGUgcHJvZ3Jlc3Npb24gb2YgdGhlIHByb2plY3QgYW5kIGZvciBjb21wbGV0ZSB0cmFuc3BhcmVuY3kuICAgDQogICAgDQoqKlNVTU1BUlkqKiAgDQpUaGlzIFIgY29kZSBpcyB1c2VkIHRvIGVzdGltYXRlIHRoZSByZWxhdGlvbnNoaXAgYmV0d2VlbiBveHlnZW4gY29uc3VtcHRpb24gKE1P4oKCKSBhbmQgYW1iaWVudCBveHlnZW4gcGFydGlhbCBwcmVzc3VyZSAoUE/igoIpIGluIHRoZSBDb21tb24gR2FsYXhpYXMgKCpHYWxheGlhcyBtYWN1bGF0dXMqKS4gSXQgYWxzbyBlc3RpbWF0ZXMgdGhlIGNyaXRpY2FsIHBhcnRpYWwgcHJlc3N1cmUgb2Ygb3h5Z2VuIGZvciBhZXJvYmljIG1ldGFib2xpc20gKFBjcml0KSwgd2hpY2ggaXMgY29tbW9ubHkgdW5kZXJzdG9vZCBhcyB0aGUgdGhyZXNob2xkIGJlbG93IHdoaWNoIHRoZSBveHlnZW4gY29uc3VtcHRpb24gcmF0ZSBjYW4gbm8gbG9uZ2VyIGJlIHN1c3RhaW5lZC4gVGhlIGFzc29jaWF0ZWQgYXJ0aWNsZSBpcyAiVGhlIHJvbGUgb2Ygb3Ntb3Jlc3BpcmF0b3J5IGNvbXByb21pc2UgaW4gaHlwb3hpYSB0b2xlcmFuY2Ugb2YgdGhlIHB1cnBvcnRlZGx5IG94eWNvbmZvcm1pbmcgdGVsZW9zdCAqR2FsYXhpYXMgbWFjdWxhdHVzKi4iIElmIHlvdSBjbGljayB0aGUgYENvZGVgIGJ1dHRvbiBpbiB0aGUgdG9wIHJpZ2h0IG9mIHRoZSBIVE1MIGRvY3VtZW50LCB5b3UgY2FuIGRvd25sb2FkIHRoZSBgLlJtZGAgZmlsZS4gIA0KDQoqKkFJTSoqICANClRoZSBhcnRpY2xlIGFpbXMgdG8gdGVzdCB3aGV0aGVyIEdhbGF4aWFzIG1hY3VsYXR1cyBjYW4gbWFpbnRhaW4gb3h5Z2VuIGNvbnN1bXB0aW9uIChNT+KCgikgYXMgYW1iaWVudCBQT+KCgiBmYWxscyBhbmQsIGlmIHNvLCBhdCB3aGF0IGxldmVsIGl0IHJlYWNoZXMgdGhlIGNyaXRpY2FsIHBhcnRpYWwgcHJlc3N1cmUgb2Ygb3h5Z2VuIGZvciBhZXJvYmljIG1ldGFib2xpc20gKFBjcml0KS4gIA0KDQoqKkFVVEhPUlMqKiAgDQpUaW1vdGh5IEQuIENsYXJrIF5bYV1eICANCkx1aXMgTC4gS3VjaGVubcO8bGxlciBeW2FdXiAgDQpFbGl6YWJldGggQy4gSG9vdHMgXlthXV4gIA0KTWFyeWFuZSBHcmFkaXRvIF5bYV1eICANCkpha2UgTS4gTWFydGluIF5bYSxiXV4gIA0KKkluIG5vIHBhcnRpY3VsYXIgb3JkZXIqDQoNCioqQUZGSUxJQVRJT05TKiogIA0KW2FdIFNjaG9vbCBvZiBMaWZlIGFuZCBFbnZpcm9ubWVudGFsIFNjaWVuY2VzLCBEZWFraW4gVW5pdmVyc2l0eSwgR2VlbG9uZywgVklDLCBBdXN0cmFsaWEgICAgDQpbYl0gU2Nob29sIG9mIEJpb2xvZ2ljYWwgU2NpZW5jZXMsIE1vbmFzaCBVbml2ZXJzaXR5LCBDbGF5dG9uLCBWaWN0b3JpYSwgQXVzdHJhbGlhICANCg0KKipDT05UUklCVVRPUiBST0xFUyoqICANCvCfmqcgKioqVG8gYmUgYWRkZWQqKiog8J+apw0KKkJhc2VkIG9uIHRoZSBDb250cmlidXRvciBSb2xlcyBUYXhvbm9teSAoQ1JlZGlUKSoNCg0KKipESVNDTEFJTUVSKiogIA0KSSAoSmFrZSBNYXJ0aW4pIGFtIGR5c2xleGljLiBJIGhhdmUgbWFkZSBhbiBlZmZvcnQgdG8gcmV2aWV3IHRoZSBzY3JpcHQgZm9yIGdyYW1tYXRpY2FsIGVycm9ycywgYnV0IHNvbWUgd2lsbCBsaWtlbHkgcmVtYWluLiBJIGFwb2xvZ2lzZS4gUGxlYXNlIHJlYWNoIG91dCB2aWEgdGhlIGNvbnRhY3QgZGV0YWlscyBiZWxvdyBpZiBhbnl0aGluZyBpcyB1bmNsZWFyLiAgDQoNCjwhLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLT4NCiMg8J+TpyBDb250YWN0DQo8IS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0+DQoNCioqSmFrZSBNLiBNYXJ0aW4qKg0KICANCvCfk6cgKipFbWFpbDoqKiBbamFrZS5tYXJ0aW5AZGVha2luLmVkdS5hdV0obWFpbHRvOmpha2UubWFydGluQGRlYWtpbi5lZHUuYXUpICANCiAgDQrwn5OnICoqQWx0IEVtYWlsOioqIFtqYWtlLm1hcnRpbi5yZXNlYXJjaEBnbWFpbC5jb21dKG1haWx0bzpqYWtlLm1hcnRpbi5yZXNlYXJjaEBnbWFpbC5jb20pIA0KICANCvCfjJAgKipXZWI6KiogW2pha2VtYXJ0aW4ub3JnXShodHRwczovL2pha2VtYXJ0aW4ub3JnLykgIA0KICANCvCfkJkgKipHaXRIdWIqKjogW0pha2VNYXJ0aW5SZXNlYXJjaF0oaHR0cHM6Ly9naXRodWIuY29tL0pha2VNYXJ0aW5SZXNlYXJjaCkgIA0KDQo8IS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0+DQojIOKame+4jyBLbml0IHNldHRpbmdzIA0KPCEtLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tPg0KDQpIZXJlIHdlIGRlZmluZSBvdXIgS25pdCBzZXR0aW5ncywgdG8gbWFrZSB0aGUgb3V0cHV0IG1vcmUgdXNlciBmcmllbmRseSwgYW5kIHRvIGNhY2hlIG91dHB1dCBmb3IgZmFzdGVyIGtuaXR0aW5nLiAgDQoNCmBgYHtyIHNldHVwfQ0KI2tuaXRlciBzZWV0dGluZw0Ka25pdHI6Om9wdHNfY2h1bmskc2V0KA0KbWVzc2FnZSA9IEZBTFNFLA0Kd2FybmluZyA9IEZBTFNFLCAjIG5vIHdhcm5pbmdzDQpjYWNoZSA9IFRSVUUsIyBDYWNoZWluZyB0byBzYXZlIHRpbWUgd2hlbiBrbml0aW5nDQp0aWR5ID0gVFJVRQ0KKQ0KYGBgDQoNCjwhLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLT4NCiMg8J+TpiBSZXF1aXJlZCBwYWNrYWdlcw0KPCEtLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tPg0KDQpUaGVzZSBhcmUgdGhlIFIgcGFja2FnZXMgcmVxdWlyZWQgZm9yIHRoaXMgc2NyaXB0LiBZb3Ugd2lsbCBuZWVkIHRvIGluc3RhbGwgYSBwYWNrYWdlIGNhbGxlZCBwYWNtYW4gdG8gcnVuIHRoZSBwX2xvYWQgZnVuY3Rpb24uICAgDQoNCmBgYHtyLCBtZXNzYWdlPUZBTFNFLCByZXN1bHRzPSdoaWRlJ30NCiMgeW91IG5lZWQgdG8gaW5zdGFsbCBwYWNtYW4gZmlyc3QgaW5zdGFsbC5wYWNrYWdlcygicGFjbWFuIikNCnBhY21hbjo6cF9sb2FkKCJnZ3Bsb3QyIiwgDQogICAgICAgICAgICAgICAiZ2d0aGVtZXMiLCANCiAgICAgICAgICAgICAgICJnZ2ZvcnRpZnkiLCANCiAgICAgICAgICAgICAgICJndEV4dHJhcyIsIA0KICAgICAgICAgICAgICAgImlncmFwaCIsDQogICAgICAgICAgICAgICAiZGFnaXR0eSIsDQogICAgICAgICAgICAgICAiZ2dkYWciLA0KICAgICAgICAgICAgICAgImdncmlkZ2VzIiwNCiAgICAgICAgICAgICAgICJnZ2hhbHZlcyIsDQogICAgICAgICAgICAgICAiZ2dFeHRyYSIsDQogICAgICAgICAgICAgICAiZ3JpZEV4dHJhIiwNCiAgICAgICAgICAgICAgICJjb3JycGxvdCIsDQogICAgICAgICAgICAgICAiUkNvbG9yQnJld2VyIiwgDQogICAgICAgICAgICAgICAiZ3QiLCANCiAgICAgICAgICAgICAgICJndHN1bW1hcnkiLA0KICAgICAgICAgICAgICAgImdyaWQiLA0KICAgICAgICAgICAgICAgInBsb3RseSIsDQogICAgICAgICAgICAgICAiYmF5ZXNwbG90IiwgIyBkYXRhIHZpc3VhbGlzYXRpb24NCiAgICAgICAgICAgICAgIA0KICAgICAgICAgICAgICAgICJ0aWR5dmVyc2UiLCANCiAgICAgICAgICAgICAgICJqYW5pdG9yIiwgDQogICAgICAgICAgICAgICAicmVhZHhsIiwgDQogICAgICAgICAgICAgICAiYnJvb20ubWl4ZWQiLCANCiAgICAgICAgICAgICAgICJkYXRhLnRhYmxlIiwgDQogICAgICAgICAgICAgICAiZGV2dG9vbHMiLA0KICAgICAgICAgICAgICAgImhtcyIsICMgZGF0YSB0aWR5DQogICAgICAgICAgICAgICANCiAgICAgICAgICAgICAgICJtYXJnaW5hbGVmZmVjdHMiLCANCiAgICAgICAgICAgICAgICJicm1zIiwgDQogICAgICAgICAgICAgICAicnN0YW4iLCANCiAgICAgICAgICAgICAgICJwZXJmb3JtYW5jZSIsIA0KICAgICAgICAgICAgICAgImVtbWVhbnMiLCANCiAgICAgICAgICAgICAgICJ0aWR5YmF5ZXMiLCANCiAgICAgICAgICAgICAgICJ2ZWdhbiIsDQogICAgICAgICAgICAgICAiYmV0YXJlZyIsDQogICAgICAgICAgICAgICAibG1lNCIsIA0KICAgICAgICAgICAgICAgImNhciIsIA0KICAgICAgICAgICAgICAgImxtZXJUZXN0IiwNCiAgICAgICAgICAgICAgICJxcXBsb3RyIiwNCiAgICAgICAgICAgICAgICJyZXNwaXJvbWV0cnkiLA0KICAgICAgICAgICAgICAgIm1jbHVzdCIsDQogICAgICAgICAgICAgICAiZnVycnIiLCAjIG1vZGVsbGluZyANCiAgICAgICAgICAgICAgDQogICAgICAgICAgICAgICANCiAgICAgICAgICAgICAgICJkYXRhd2l6YXJkIiwgDQogICAgICAgICAgICAgICAiU1JTIiwgIyBkYXRhIG1hbmlwdWxhdGlvbiANCiAgICAgICAgICAgICAgIA0KICAgICAgICAgICAgICAgInNlc3Npb25pbmZvIiwNCiAgICAgICAgICAgICAgICJwYW5kZXIiICMgcmVwcm9kdWNpYmlsaXR5DQogICAgICAgICAgICAgICAgICAgICAgICkNCmBgYA0KDQoNCjwhLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLT4NCiMg8J+UpyBDdXN0b20gZnVuY3Rpb25zIA0KPCEtLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tPg0KDQpIZXJlIGFyZSBzb21lIGN1c3RvbSBmdW5jdGlvbiB1c2VkIHdpdGhpbiB0aGlzIHNjcmlwdC4gICAgDQoNCmBiYXllc19pbmNyZW1lbnRhbF9yZWdyZXNzaW9uX2J5X2lkKClgOiBBIGN1c3RvbSBmdW5jdGlvbiB0byBidWlsZCBCYXllc2lhbiBpbmNyZW1lbnRhbCByZWdyZXNzaW9ucy4gSXQgaXMgZm9ybWF0dGVkIGZvciBydW5uaW5nIGEgbGlzdCBvZiBzdWIgZ3JvdXAgbW9kZWxzIChpZHMpIGluIHBhcmFsbGVsIHdpdGggNCBjb3Jlcy4gSXQgdXNlcyBgYnJtKClgIHdpdGggYSBHYXVzc2lhbiBlcnJvciBmdW5jdGlvbi4gICANCg0KKipVc2UqKjogVGhlIGZ1bmN0aW9uIHRha2VzIGFuIGdyb3VwaW5nIGZhY3Rvci9pZCB0aGF0IHdpbGwgYmUgdXNlZCB0byBmaWx0ZXIgdGhlIGRhdGEgZm9yIHRoZSByZWdyZXNzaW9uLCBpZiBub25lIGlzIHByb3ZlZCBpdCB3aWxsIHVzZSBhbGwgZGF0YSAoaWRfaSksIHRoZSBjb2x1bW4gbmFtZSBvZiB0aGUgZ3JvdXBpbmcgZmFjdG9yL2NoYXJ0ZXIgaW4gdGhlIGRhdGEgZnJhbWUsIGlmIG5vbmUgaXMgcHJvdmVkIGl0IHdpbGwgdXNlIGFsbCBkYXRhIChpZF9uYW1lKSwgdGhlIGRhdGEgZnJhbWUgKGRhdGEpLCB0aGUgcHJlZGljdG9yIG9mIGludGVyZXN0IChwcmVkaWN0b3IpLCB0aGUgcmVzcG9uc2Ugb2YgaW50ZXJlc3QgKHJlc3BvbnNlKSwgdGhlIHJhbmRvbSBzZWVkIG51bWJlciBmb3IgbW9kZWwgcmVwcm9kdWNpYmlsaXR5IChzZWVkX251bWJlciksIHdoZXJlIHlvdSB3b3VsZCBsaWtlIHRvIHNhdmUgdGhlIG1vZGVsIG91dHB1dHMgKHNhdmVfbW9kZWxzID0gbG9naWNhbCBhcmd1bWVudCksIHRoZSBvdXRwdXQgZGlyZWN0b3J5IGZvciB0aGUgc2F2ZWQgcmRzIG1vZGVscyAobW9kX291dHB1dF93ZCkgICANCg0KYmF5ZXNfaW5jcmVtZW50YWxfcmVncmVzc2lvbl9ieV9pZChpZF9pID0gInlvdXJfaWQiLCBpZF9uYW1lID0gImNvbHVtbiBuYW1lIGZvciBpZHMiLCBkYXRhID0gInlvdXIgZGF0YSIsIHByZWRpY3RvciA9ICJwcmVkaWN0b3IgdmFyaWFibGUiLCByZXNwb25zZSA9ICJyZXNwb25zZSB2YXJpYWJsZSIsIHNlZWRfbnVtYmVyID0gaW50ZWdlciwgc2F2ZV9tb2RlbHMgPSBUUlVFL0ZBTFNFLCBtb2Rfb3V0cHV0X3dkID0gImRpcmVjdG9yeSBmb3IgbW9kZWwgb3V0cHV0IikNCg0KYGBge3J9DQojIEluc3RlYWQgd2UgY291bGQgYWxzbyB1c2UgYSBkaXN0cmlidXRpb25hbCByZWdyZXNzaW9uIGFwcHJvYWNoLCBieSBzcGVjaWZpY2FsbHkgbW9kZWxsaW5nIHRoZSB2YXJpYW5jZSBieSBETyAoZS5nLiBzaWdtYSB+IERPKS4gV2VpZ2h0aW5nIG1heSBub3QgYmUgcmVxdWlyZWQgaW4gdGhpcyBjYXNlLCBJIGRvbid0IHRoaW5rIGhpZ2hlciBkZW5zaXR5IG9mIHZhdWxlcyBpbiBhIGdpdmVuIHNwYWNlIHdpbGwgZWZmZWN0IEJheWVzaWFuIGVzdGltYXRlcyBsaWtlIGl0IGRvZXMgaW4gZnJlcXVlbnRpc3QgbW9kZWxzLiBTZWUgZGlzY291cnNlIGh0dHBzOi8vZGlzY291cnNlLm1jLXN0YW4ub3JnL3Qvd2VpZ2h0cy1pbi1icm0vNDI3OCAgIA0KDQpiYXllc19pbmNyZW1lbnRhbF9yZWdyZXNzaW9uX2J5X2lkIDwtIGZ1bmN0aW9uKGlkX2ksIGlkX25hbWUsIGRhdGEsIHByZWRpY3RvciwgcmVzcG9uc2UsIHNlZWRfbnVtYmVyLCBzYXZlX21vZGVscywgbW9kX291dHB1dF93ZCkgew0KICAjIEluaXRpYXRlIGFuIGVtcHR5IGxpc3QgdG8gc3RvcmUgbW9kZWxzDQogIG1vZGVscyA8LSBsaXN0KCkNCiAgDQogICMgQ2hlY2sgaWYgaWRfbmFtZSBpcyBtaXNzaW5nLCBOVUxMLCBvciBibGFuaywgYW5kIGFzc2lnbiBOQSBpZiBzbw0KICBpZiAobWlzc2luZyhpZF9uYW1lKSB8fCBpcy5udWxsKGlkX25hbWUpIHx8IGlkX25hbWUgPT0gIiIpIHsNCiAgICBpZF9uYW1lIDwtIE5BDQogIH0NCiAgDQogICMgQ2hlY2sgaWYgaWRfaSBpcyBtaXNzaW5nLCBOVUxMLCBvciBibGFuaywgYW5kIGFzc2lnbiBOQSBpZiBzbw0KICBpZiAobWlzc2luZyhpZF9pKSB8fCBpcy5udWxsKGlkX2kpIHx8IGlkX2kgPT0gIiIpIHsNCiAgICBpZF9uYW1lIDwtIE5BDQogIH0NCiAgDQogICMgRmlsdGVyIGRhdGEgZm9yIHRoZSBjdXJyZW50IElEIGlmIGlkX25hbWUgaXMgZ2l2ZW4gYXMgYSBmYWN0b3Igb3IgY2hhcmFjdGVyIGFuZCBpZF9pIGlzIGRlZmluZWQNCiAgZGZfaSA8LSBkYXRhICU+JQ0KICAgIGRwbHlyOjpmaWx0ZXIoDQogICAgICBpZiAoIWlzLm5hKGlkX2kpICYmIChpcy5mYWN0b3IoZGF0YVtbaWRfbmFtZV1dKSB8fCBpcy5jaGFyYWN0ZXIoZGF0YVtbaWRfbmFtZV1dKSkpIHsNCiAgICAgICAgISFybGFuZzo6c3ltKGlkX25hbWUpID09IGlkX2kNCiAgICAgIH0gZWxzZSB7DQogICAgICAgIFRSVUUNCiAgICAgIH0NCiAgICApDQogIA0KICAjIER5bmFtaWNhbGx5IGNyZWF0ZSBmb3JtdWxhcw0KICBmb3JtdWxhX2xtXzAgPC0gcmVmb3JtdWxhdGUoIjEiLCByZXNwb25zZSkNCiAgZm9ybXVsYV9sbV8xIDwtIHJlZm9ybXVsYXRlKHByZWRpY3RvciwgcmVzcG9uc2UpDQogIGZvcm11bGFfbG1fMiA8LSByZWZvcm11bGF0ZShzcHJpbnRmKCJwb2x5KCVzLCAyKSIsIHByZWRpY3RvciksIHJlc3BvbnNlKQ0KICBmb3JtdWxhX2xtXzMgPC0gcmVmb3JtdWxhdGUoc3ByaW50ZigicG9seSglcywgMykiLCBwcmVkaWN0b3IpLCByZXNwb25zZSkNCiAgDQogICMgRml0IGFuZCBzdG9yZSBtb2RlbHMgaW4gdGhlIGxpc3QNCiAgbW9kZWxzW1twYXN0ZTAoaWRfaSwgIl9sbV8wIildXSA8LSBicm0oDQogICAgYmYoZm9ybXVsYV9sbV8wLCBmYW1pbHkgPSBnYXVzc2lhbigpKSwgDQogICAgZGF0YSA9IGRmX2ksIGNvcmVzID0gNCwgc2VlZCA9IHNlZWRfbnVtYmVyLCBzYXZlX3BhcnMgPSBzYXZlX3BhcnMoYWxsID0gc2F2ZV9tb2RlbHMpLA0KICAgIHNhbXBsZV9wcmlvciA9IEZBTFNFLCBzaWxlbnQgPSBUUlVFLCBmaWxlID0gcGFzdGUwKG1vZF9vdXRwdXRfd2QsICIvIiwgaWRfaSwgIl9sbV8wIikNCiAgKQ0KICANCiAgbW9kZWxzW1twYXN0ZTAoaWRfaSwgIl9sbV8xIildXSA8LSBicm0oDQogICAgYmYoZm9ybXVsYV9sbV8xLCBmYW1pbHkgPSBnYXVzc2lhbigpKSwgDQogICAgZGF0YSA9IGRmX2ksIGNvcmVzID0gNCwgc2VlZCA9IHNlZWRfbnVtYmVyLCBzYXZlX3BhcnMgPSBzYXZlX3BhcnMoYWxsID0gc2F2ZV9tb2RlbHMpLA0KICAgIHNhbXBsZV9wcmlvciA9IEZBTFNFLCBzaWxlbnQgPSBUUlVFLCBmaWxlID0gcGFzdGUwKG1vZF9vdXRwdXRfd2QsICIvIiwgaWRfaSwgIl9sbV8xIikNCiAgKQ0KICANCiAgbW9kZWxzW1twYXN0ZTAoaWRfaSwgIl9sbV8yIildXSA8LSBicm0oDQogICAgYmYoZm9ybXVsYV9sbV8yLCBmYW1pbHkgPSBnYXVzc2lhbigpKSwgDQogICAgZGF0YSA9IGRmX2ksIGNvcmVzID0gNCwgc2VlZCA9IHNlZWRfbnVtYmVyLCBzYXZlX3BhcnMgPSBzYXZlX3BhcnMoYWxsID0gc2F2ZV9tb2RlbHMpLA0KICAgIHNhbXBsZV9wcmlvciA9IEZBTFNFLCBzaWxlbnQgPSBUUlVFLCBmaWxlID0gcGFzdGUwKG1vZF9vdXRwdXRfd2QsICIvIiwgaWRfaSwgIl9sbV8yIikNCiAgKQ0KICANCiAgbW9kZWxzW1twYXN0ZTAoaWRfaSwgIl9sbV8zIildXSA8LSBicm0oDQogICAgYmYoZm9ybXVsYV9sbV8zLCBmYW1pbHkgPSBnYXVzc2lhbigpKSwgDQogICAgZGF0YSA9IGRmX2ksIGNvcmVzID0gNCwgc2VlZCA9IDE0MzAxOSwgc2F2ZV9wYXJzID0gc2F2ZV9wYXJzKGFsbCA9IHNhdmVfbW9kZWxzKSwNCiAgICBzYW1wbGVfcHJpb3IgPSBGQUxTRSwgc2lsZW50ID0gVFJVRSwgZmlsZSA9IHBhc3RlMChtb2Rfb3V0cHV0X3dkLCAiLyIsIGlkX2ksICJfbG1fMyIpDQogICkNCiAgDQogICMgUmV0dXJuIHRoZSBsaXN0IG9mIG1vZGVscyBmb3IgdGhlIGN1cnJlbnQgSUQNCiAgcmV0dXJuKG1vZGVscykNCn0NCmBgYA0KDQpgbG9hZF9yZHM5KClgOiBBIGN1c3RvbSBmdW5jdGlvbiB0byBsb2FkIGFsbCByZHMgbW9kZWxzIGluIGEgZGlyZWN0b3J5IGFuZCBzdG9yZSBpbiBhIGxpc3QuIFRha2VzIGEgZGlyZWN0b3J5IHdpdGggYC5yZHNgIGZpbGVzDQoNCmBgYHtyfQ0KbG9hZF9yZHMgPC0gZnVuY3Rpb24obW9kZWxfZHcpIHsNCiAgIyBMaXN0IGFsbCAucmRzIGZpbGVzIGluIHRoZSBkaXJlY3RvcnkNCiAgbW9kZWxfZmlsZV9saXN0IDwtIGxpc3QuZmlsZXMocGF0aCA9IG1vZGVsX2R3LCBwYXR0ZXJuID0gIlxcLnJkcyQiLCBmdWxsLm5hbWVzID0gVFJVRSkNCiAgDQogICMgSW5pdGlhbGlzZSBhbiBlbXB0eSBsaXN0IHRvIHN0b3JlIG1vZGVscw0KICBtb2RlbF9zdG9yZV9saXN0IDwtIGxpc3QoKQ0KICANCiAgIyBJdGVyYXRlIHRocm91Z2ggZWFjaCBmaWxlIGFuZCBsb2FkIHRoZSBSRFMNCiAgZm9yIChtb2RfaSBpbiBtb2RlbF9maWxlX2xpc3QpIHsNCiAgICBtb2QgPC0gcmVhZFJEUyhmaWxlID0gbW9kX2kpICAjIFJlYWQgdGhlIFJEUyBmaWxlDQogICAgbW9kZWxfbmFtZSA8LSB0b29sczo6ZmlsZV9wYXRoX3NhbnNfZXh0KGJhc2VuYW1lKG1vZF9pKSkgICMgRXh0cmFjdCB0aGUgZmlsZSBuYW1lIHdpdGhvdXQgZXh0ZW5zaW9uDQogICAgbW9kZWxfc3RvcmVfbGlzdFtbbW9kZWxfbmFtZV1dIDwtIG1vZCAgIyBTdG9yZSBpdCBpbiB0aGUgbGlzdA0KICB9DQogIA0KICAjIFJldHVybiB0aGUgbGlzdCBvZiBtb2RlbHMNCiAgcmV0dXJuKG1vZGVsX3N0b3JlX2xpc3QpDQp9DQpgYGANCg0KDQpgaW5jcmVtZW50YWxfcmVncmVzc2lvbl9iYXllc19maXRzKClgOiBBIGN1c3RvbSBmdW5jdGlvbiBmb3IgcHVsbGluZyBtb2RlbCBmaXRzLCBsb28gYW5kIHIyIHVzaW5nIGBsb28oKWAgYW5kIGBiYXllc19SMigpYCwgcmVzcGVjdGl2ZWx5LiBUYWtlcyBhIGxpc3Qgb2YgYGJybWAgbW9kZWxzLiAgIA0KDQpgYGB7cn0NCiMgRGVmaW5lIEZ1bmN0aW9uIHRvIFByb2Nlc3MgdGhlIGRhdGEgZm9yIGVhY2ggSUQNCmluY3JlbWVudGFsX3JlZ3Jlc3Npb25fYmF5ZXNfZml0cyA8LSBmdW5jdGlvbihtb2RlbHMpIHsNCiAgDQogIGxvb19yZXN1bHRzX2xpc3QgPC0gbGlzdCgpDQogIA0KICAjIEl0ZXJhdGUgb3ZlciB0aGUgbmFtZXMgb2YgdGhlIG1vZGVscw0KICBmb3IgKG1vZF9uYW1lIGluIG5hbWVzKG1vZGVscykpIHsNCiAgICAjIEV4dHJhY3QgdGhlIG1vZGVsDQogICAgbW9kX2kgPC0gbW9kZWxzW1ttb2RfbmFtZV1dDQogICAgDQogICAgIyBDb21wdXRlIExPTyByZXN1bHRzDQogICAgbW9kX2xvb19yZXN1bHRzX2kgPC0gbG9vOjpsb28obW9kX2kpDQogICAgDQogICAgIyBFeHRyYWN0IHJlbGV2YW50IExPTyBtZXRyaWNzDQogICAgZWxwZF9sb29faSA8LSBtb2RfbG9vX3Jlc3VsdHNfaSRlbHBkX2xvbw0KICAgIHBfbG9vX2kgPC0gbW9kX2xvb19yZXN1bHRzX2kkcF9sb28NCiAgICBsb29pY19pIDwtIG1vZF9sb29fcmVzdWx0c19pJGxvb2ljDQogICAgDQogICAgIyBDcmVhdGUgYSBkYXRhIGZyYW1lIHdpdGggbWV0cmljcw0KICAgIGRmX2kgPC0gZGF0YS5mcmFtZSgNCiAgICAgIGVscGRfbG9vID0gZWxwZF9sb29faSwNCiAgICAgIHBfbG9vID0gcF9sb29faSwNCiAgICAgIGxvb2ljID0gbG9vaWNfaSwNCiAgICAgIG1vZGVsID0gbW9kX25hbWUNCiAgICApDQogICAgDQogICAgZXN0X2kgPC0gdGlkeShtb2RfaSwgZWZmZWN0cyA9ICJmaXhlZCIsIGNvbmYuaW50ID0gVFJVRSkgJT4lIA0KICAgICAgZHBseXI6OnNlbGVjdCh0ZXJtLCBlc3RpbWF0ZSwgY29uZi5sb3csIGNvbmYuaGlnaCkgJT4lIA0KICAgICAgdGlkeXI6OnBpdm90X3dpZGVyKA0KICAgICAgICBuYW1lc19mcm9tID0gdGVybSwgICAgICAgICAgICAgICAgIyBVc2UgYHRlcm1gIGFzIGNvbHVtbiBuYW1lcw0KICAgICAgICB2YWx1ZXNfZnJvbSA9IGMoZXN0aW1hdGUsIGNvbmYubG93LCBjb25mLmhpZ2gpLCAgIyBWYWx1ZXMgdG8gcGl2b3QNCiAgICAgICAgbmFtZXNfc2VwID0gIl8iICAgICAgICAgICAgICAgICAgICMgQWRkIGEgc2VwYXJhdG9yIHRvIGNvbHVtbiBuYW1lcw0KICAgICAgKQ0KICAgIA0KICAgIGRmX2kgPC0gY2JpbmQoZGZfaSwgZXN0X2kpDQogICAgDQogICAgIyBTdG9yZSB0aGUgZGF0YSBmcmFtZSBpbiB0aGUgbGlzdA0KICAgIGxvb19yZXN1bHRzX2xpc3RbW21vZF9uYW1lXV0gPC0gZGZfaQ0KICB9DQogIA0KICAjIENvbWJpbmQgDQogIGxvb19yZXN1bHRzX2NvbWJpbmVkIDwtIGJpbmRfcm93cyhsb29fcmVzdWx0c19saXN0KQ0KICANCiAgIyBHZXQgUjIgDQogIHIyX3Jlc3VsdHMgPC0gbWFwX2Rmcihtb2RlbHMsIH4gYXMuZGF0YS5mcmFtZShiYXllc19SMigueCkpLCAuaWQgPSAibW9kZWwiKSAlPiUNCiAgICB0aWJibGU6OnJlbW92ZV9yb3duYW1lcygpDQogIA0KICAjIENvbWJpbmQgUjIgYW5kIGxvbyByZXN1bHRzIA0KICBtb2RlbF9maXRfZGYgPC0gZHBseXI6OmZ1bGxfam9pbihsb29fcmVzdWx0c19jb21iaW5lZCwgcjJfcmVzdWx0cywgYnkgPSAibW9kZWwiKSAlPiUgDQogICAgZHBseXI6OnNlbGVjdChtb2RlbCwgZXZlcnl0aGluZygpKSAlPiUgDQogICAgZHBseXI6OnJlbmFtZShyMiA9IEVzdGltYXRlLA0KICAgICAgICAgICAgICAgICAgcjJfZXJyb3IgPSBFc3QuRXJyb3IsDQogICAgICAgICAgICAgICAgICByMl9xMi41ID0gUTIuNSwNCiAgICAgICAgICAgICAgICAgIHIyX3E5Ny41ID0gUTk3LjUpICU+JSANCiAgICBkcGx5cjo6bXV0YXRlKGlkID0gc3ViKCJfKGxtX1xcZCspJCIsICIiLCBtb2RlbCksDQogICAgICAgICAgICAgICAgICBtb2RlbF90eXBlID0gc3ViKCJeLipfKGxtX1xcZCspJCIsICJcXDEiLCBtb2RlbCkpDQogIA0KICByZXR1cm4obW9kZWxfZml0X2RmKQ0KfQ0KYGBgDQoNCmBiYXllc19tb2RfcHJlZGljdGlvbnMoKWA6IFRoaXMgZnVuY3Rpb24gZXh0cmFjdHMgdGhlIHByZWRpY3RlZCB2YWx1ZXMgdXNpbmcgYGZpdHRlZCgpYCBmcm9tIGEgbGlzdCBvZiBtb2RlbHMgYW5kIGNvbWJpbmVzIHRoZW0gd2l0aCB0aGUgb3JpZ2luYWwgZGF0YSBmaWxlIHVzZWQgZm9yIHRoZSBtb2RlbC4gVGhlc2UgYXJlIHRoZSAqKnBvc3RlcmlvciBtZWFuIGZpdHRlZCB2YWx1ZXMqKiAoaS5lLiB0aGUgZXhwZWN0ZWQgdmFsdWUgb2YgdGhlIHJlc3BvbnNlIHZhcmlhYmxlIGdpdmVuIHRoZSBwcmVkaWN0b3IgdmFyaWFibGVzIGFuZCB0aGUgZXN0aW1hdGVkIHBvc3RlcmlvciBkaXN0cmlidXRpb25zIG9mIHRoZSBwYXJhbWV0ZXJzKSBmb3IgZWFjaCBvYnNlcnZhdGlvbiBpbiB0aGUgZGF0YXNldCwgYWxvbmcgd2l0aCAqKjk1JSBjcmVkaWJsZSBpbnRlcnZhbHMqKi4NCg0KYGBge3J9DQpiYXllc19tb2RfcHJlZGljdGlvbnMgPC0gZnVuY3Rpb24obW9kZWxzLCBvcmlnaW5hbF9kYXRhKSB7DQogIA0KICBwcmVkaWN0aW9uX2xpc3QgPC0gbGlzdCgpDQogIA0KICBmb3IgKG1vZF9uYW1lIGluIG5hbWVzKG1vZGVscykpIHsNCiAgICAjIEV4dHJhY3QgbW9kDQogICAgbW9kX2kgPC0gbW9kZWxzW1ttb2RfbmFtZV1dDQogICAgDQogICAgIyBNYWtlIG1vZGUgcHJlZGljdGlvbnMNCiAgICBtb2RlbF9wcmVkaWN0aW9uc19pIDwtIGFzLmRhdGEuZnJhbWUoZml0dGVkKG1vZF9pLCBzdW1tYXJ5ID0gVFJVRSkpICU+JSANCiAgICAgIGRwbHlyOjptdXRhdGUobW9kZWwgPSBtb2RfbmFtZSwNCiAgICAgICAgICAgICAgICAgICAgaWQgPSBzdWIoIl8obG1fXFxkKykkIiwgIiIsIG1vZF9uYW1lKSwNCiAgICAgICAgICAgICAgICAgICAgbW9kZWxfdHlwZSA9IHN1YigiXi4qXyhsbV9cXGQrKSQiLCAiXFwxIiwgbW9kX25hbWUpKSAlPiUgDQogICAgICBkcGx5cjo6cmVuYW1lKHByZWRfbG93ZXIgPSBRMi41LCBwcmVkX3VwcGVyID0gUTk3LjUsIHByZWRpY3RlZCA9IEVzdGltYXRlLCBwcmVkX2Vycm9yID0gRXN0LkVycm9yKSAlPiUgDQogICAgICBkcGx5cjo6c2VsZWN0KG1vZGVsLCBldmVyeXRoaW5nKCkpDQogICAgDQogICAgaWRfaSA8LSBtb2RlbF9wcmVkaWN0aW9uc19pJGlkWzFdDQogICAgDQogICAgb3JpZ2luYWxfZGF0YV9pIDwtIG9yaWdpbmFsX2RhdGEgJT4lIA0KICAgICAgZHBseXI6OmZpbHRlcihpZCA9PSBpZF9pKSAlPiUgDQogICAgICBkcGx5cjo6c2VsZWN0KC1pZCkNCiAgICANCiAgICBtb2RlbF9wcmVkaWN0aW9uc19vcmlnaW5hbF9pIDwtIGNiaW5kKG1vZGVsX3ByZWRpY3Rpb25zX2ksIG9yaWdpbmFsX2RhdGFfaSkNCiAgICANCiAgICBwcmVkaWN0aW9uX2xpc3RbW21vZF9uYW1lXV0gPC0gbW9kZWxfcHJlZGljdGlvbnNfb3JpZ2luYWxfaQ0KICB9DQogIHByZWRpY3Rpb25zX2RmIDwtIGJpbmRfcm93cyhwcmVkaWN0aW9uX2xpc3QpIA0KICByZXR1cm4ocHJlZGljdGlvbnNfZGYpDQp9DQpgYGANCg0KDQpgY2FsY1NNUigpYDogKiphdXRob3JlZCBieSBEZW5pcyBDaGFib3QqKiB1c2VkIHRvIGVzdGltYXRlIFNNUiB3aXRoIHNldmVyYWwgZGlmZmVyZW50IG1ldGhvZHMgQ2xhaXJlYXV4IGFuZCBDaGFib3QgKDIwMTYpIF5bMV1eICAgDQoNCl5bMV1eIENsYWlyZWF1eCwgRy4gYW5kIENoYWJvdCwgRC4gKDIwMTYpIFJlc3BvbnNlcyBieSBmaXNoZXMgdG8gZW52aXJvbm1lbnRhbCBoeXBveGlhOiBpbnRlZ3JhdGlvbiB0aHJvdWdoIEZyeSdzIGNvbmNlcHQgb2YgYWVyb2JpYyBtZXRhYm9saWMgc2NvcGUuIEpvdXJuYWwgb2YgRmlzaCBCaW9sb2d5IGh0dHBzOi8vZG9pLm9yZy8xMC4xMTExL2pmYi4xMjgzMyAgIA0KDQpgYGB7cn0NCmNhbGNTTVIgPSBmdW5jdGlvbihZLCBxPWMoMC4xLDAuMTUsMC4yLDAuMjUsMC4zKSwgRz0xOjQpew0KCXUgPSBzb3J0KFkpDQoJdGhlLk1jbHVzdCA8LSBNY2x1c3QoWSwgIEc9RykNCgljbCA8LSB0aGUuTWNsdXN0JGNsYXNzaWZpY2F0aW9uDQoJIyBzb21ldGltZXMsIHRoZSBjbGFzcyBjb250YWluaW5nIFNNUiBpcyBub3QgY2FsbGVkIDENCgkjIHRoZSBmb2xsb3dpbmcgcHJlc3VtZXMgdGhhdCB3aGVuIGNsYXNzIDEgY29udGFpbnMgPiAxMCUgb2YgY2FzZXMsIA0KCSMgaXQgY29udGFpbnMgU01SLCBvdGhlcndpc2Ugd2UgdGFrZSBjbGFzcyAyDQoJY2wyIDwtIGFzLmRhdGEuZnJhbWUodGFibGUoY2wpKQ0KCWNsMiRjbCA8LSBhcy5udW1lcmljKGxldmVscyhjbDIkY2wpKQ0KCXZhbGlkIDwtIGNsMiRGcmVxPj0wLjEqbGVuZ3RoKHRpbWUpICANCgl0aGUuY2wgPC0gbWluKGNsMiRjbFt2YWxpZF0pDQoJbGVmdC5kaXN0ciA8LSBZW3RoZS5NY2x1c3QkY2xhc3NpZmljYXRpb249PXRoZS5jbF0NCgltbG5kID0gdGhlLk1jbHVzdCRwYXJhbWV0ZXJzJG1lYW5bdGhlLmNsXQ0KCUNWbWxuZCA9IHNkKGxlZnQuZGlzdHIpL21sbmQgKiAxMDANCglxdWFudD1xdWFudGlsZShZLCBxKQ0KCWxvdzEwPW1lYW4odVsxOjEwXSkNCglsb3cxMHBjID0gbWVhbih1WzY6KDUgKyByb3VuZCgwLjEqKGxlbmd0aCh1KS01KSkpXSkNCgkjIHJlbW92ZSA1IG91dGxpZXJzLCBrZWVwIGxvd2VzdCAxMCUgb2YgdGhlIHJlc3QsIGF2ZXJhZ2UNCgkjIEhlcnJtYW5uICYgRW5kZXJzIDIwMDANCglyZXR1cm4obGlzdChtbG5kPW1sbmQsIHF1YW50PXF1YW50LCBsb3cxMD1sb3cxMCwgbG93MTBwYz1sb3cxMHBjLA0KCQkgICAgICBjbD1jbCwgQ1ZtbG5kPUNWbWxuZCkpDQp9DQpgYGANCg0KDQpgY2FsY08yY3JpdCgpYDogKiphdXRob3JlZCBieSBEZW5pcyBDaGFib3QqKiB1c2VkIHRvIGVzdGltYXRlIE8yY3JpdCAoUGNyaXB0KS4gQ2xhaXJlYXV4IGFuZCBDaGFib3QgKDIwMTYpIF5bMV1eICANCg0KKioqTm90ZTogTzIgaXMgYXNzdW1lZCB0byBiZSBpbiBwZXJjZW50YWdlIG9mIGRpc3NvbHZlZCBveHlnZW4gKERPKSoqKg0KDQpgYGB7cn0NCmNhbGNPMmNyaXQgPC0gZnVuY3Rpb24oRGF0YSwgU01SLCBsb3dlc3RNTzIgPSBOQSwgZ2FwTGltaXQgPSA0LA0KbWF4Lm5iLk1PMi5mb3IucmVnID0gMjApDQp7DQojIEFVVEhPUjogRGVuaXMgQ2hhYm90LCBJbnN0aXR1dCBNYXVyaWNlLUxhbW9udGFnbmUsIERGTywgQ2FuYWRhDQojIGZpcnN0IHZlcnNpb24gd3JpdHRlbiBpbiBKdW5lIDIwMDkNCiMgbGFzdCB1cGRhdGVkIGluIEphbnVhcnkgMjAxNQ0KbWV0aG9kID0gIkxTX3JlZyIgIyB3aWxsIGJlY29tZSAidGhyb3VnaF9vcmlnaW4iIGlmIGludGVyY2VwdCBpcyA+IDANCmlmKGlzLm5hKGxvd2VzdE1PMikpIGxvd2VzdE1PMiA9IHF1YW50aWxlKERhdGEkTU8yW0RhdGEkRE8gPj0gODBdLCBwPTAuMDUpDQojIFN0ZXAgMTogaWRlbnRpZnkgcG9pbnRzIHdoZXJlIE1PMiBpcyBwcm9wb3J0aW9uYWwgdG8gRE8NCmdlcVNNUiA9IERhdGEkTU8yID49IGxvd2VzdE1PMg0KcGl2b3RETyA9IG1pbihEYXRhJERPW2dlcVNNUl0pDQpsZXRoYWwgPSBEYXRhJERPIDwgcGl2b3RETw0KTl91bmRlcl9TTVIgPSBzdW0obGV0aGFsKSAjIHBvaW50cyBhdmFpbGFibGUgZm9yIHJlZ3Jlc3Npb24/DQpmaW5hbF9OX3VuZGVyX1NNUiA9IGxldGhhbCAjIHNvbWUgcG9pbnRzIG1heSBiZSByZW1vdmVkIGF0IFN0ZXAgNA0KbGFzdE1PMnJlZyA9IERhdGEkTU8yW0RhdGEkRE8gPT0gcGl2b3RET10gIyBsYXN0IE1PMiB3aGVuIHJlZ3VsYXRpbmcNCmlmKE5fdW5kZXJfU01SID4gMSkgdGhlTW9kID0gbG0oTU8yfkRPLCBkYXRhPURhdGFbbGV0aGFsLF0pDQojIFN0ZXAgMiwgYWRkIG9uZSBvciBtb3JlIHBvaW50IGF0IG9yIGFib3ZlIFNNUg0KIyAyQSwgd2hlbiB0aGVyZSBhcmUgZmV3ZXIgdGhhbiAzIHZhbGlkIHBvaW50cyB0byBjYWxjdWxhdGUgYSByZWdyZXNzaW9uDQppZihOX3VuZGVyX1NNUiA8IDMpew0KbWlzc2luZyA9IDMgLSBzdW0obGV0aGFsKQ0Kbm90LmxldGhhbCA9IERhdGEkRE9bZ2VxU01SXQ0KRE9saW1pdCA9IG1heChzb3J0KG5vdC5sZXRoYWwpWzE6bWlzc2luZ10pICMgaGlnaGVzdCBETyBhY2NlcHRhYmxlDQojIHRvIHJlYWNoIGEgTiBvZiAzDQphZGRlZFBvaW50cyA9IERhdGEkRE8gPD0gRE9saW1pdA0KbGV0aGFsID0gbGV0aGFsIHwgYWRkZWRQb2ludHMNCnRoZU1vZCA9IGxtKE1PMn5ETywgZGF0YT1EYXRhW2xldGhhbCxdKQ0KfQ0KIyAyQiwgYWRkIHBpdm90RE8gdG8gdGhlIGZpdCB3aGVuIFN0ZXAgMSB5aWVsZGVkIDMgb3IgbW9yZSB2YWx1ZXM/DQppZihOX3VuZGVyX1NNUiA+PSAzKXsNCmxldGhhbEIgPSBEYXRhJERPIDw9IHBpdm90RE8gIyBoYXMgb25lIG1vcmUgdmFsdWUgdGhhbiAibGV0aGFsIg0KcmVnQSA9IHRoZU1vZA0KcmVnQiA9IGxtKE1PMn5ETywgZGF0YT1EYXRhW2xldGhhbEIsXSkNCmxhcmdlX3Nsb3BlX2Ryb3AgPSAoY29lZihyZWdBKVsyXS9jb2VmKHJlZ0IpWzJdKSA+IDEuMSAjIGFyYml0cmFyeQ0KbGFyZ2VfRE9fZ2FwID0gKG1heChEYXRhJERPW2xldGhhbEJdKSAtIG1heChEYXRhJERPW2xldGhhbF0pKSA+IGdhcExpbWl0DQp0b29TbWFsbE1PMiA9IGxhc3RNTzJyZWcgPCBTTVINCmlmKCFsYXJnZV9zbG9wZV9kcm9wICYgIWxhcmdlX0RPX2dhcCAmICF0b29TbWFsbE1PMikgew0KbGV0aGFsID0gbGV0aGFsQg0KdGhlTW9kID0gcmVnQg0KfSAjIG90aGVyd2lzZSB3ZSBkbyBub3QgYWNjZXB0IHRoZSBhZGRpdGlvbmFsIHBvaW50DQp9DQojIFN0ZXAgMw0KIyBpZiB0aGUgdXNlciB3YW50cyB0byBsaW1pdCB0aGUgbnVtYmVyIG9mIHBvaW50cyBpbiB0aGUgcmVncmVzc2lvbg0KaWYoIWlzLm5hKG1heC5uYi5NTzIuZm9yLnJlZykgJiBzdW0obGV0aGFsKT5tYXgubmIuTU8yLmZvci5yZWcpew0KUmFua3MgPSByYW5rKERhdGEkRE8pDQpsZXRoYWwgPSBSYW5rcyA8PSBtYXgubmIuTU8yLmZvci5yZWcNCnRoZU1vZCA9IGxtKE1PMn5ETywgZGF0YT1EYXRhW2xldGhhbCxdKQ0KZmluYWxfTl91bmRlcl9TTVIgPSBtYXgubmIuTU8yLmZvci5yZWcNCn0NCiMgU3RlcCA0DQpwcmVkTU8yID0gYXMubnVtZXJpYyhwcmVkaWN0KHRoZU1vZCwgZGF0YS5mcmFtZShETz1EYXRhJERPKSkpDQpEYXRhJGRlbHRhID0gKERhdGEkTU8yLXByZWRNTzIpL3ByZWRNTzIgKiAxMDAgIyByZXNpZHVhbHMgc2V0IHRvIHplcm8NCiMgd2hlbiBiZWxvdyBwaXZvdERPDQpEYXRhJGRlbHRhW0RhdGEkRE8gPCBwaXZvdERPIHwgbGV0aGFsXSA9IDANCnRvbCA9IDAgIyBhbnkgcG9zaXRpdmUgcmVzaWR1YWwgaXMgdW5hY2NlcHRhYmxlDQpIaWdoVmFsdWVzID0gRGF0YSRkZWx0YSA+IHRvbA0KUmFua3MgPSByYW5rKC0xKkRhdGEkZGVsdGEpDQpIaWdoTU8yID0gSGlnaFZhbHVlcyAmIFJhbmtzID09IG1pbihSYW5rcykgIyBrZWVwIGxhcmdlc3QgcmVzaWR1YWwNCmlmIChzdW0oSGlnaFZhbHVlcykgPiAwKSB7DQpuYmxldGhhbCA9IHN1bShsZXRoYWwpDQpEYXRhJFcgPSBOQQ0KRGF0YSRXW2xldGhhbF09MS9uYmxldGhhbA0KRGF0YSRXW0hpZ2hNTzJdID0gMQ0KdGhlTW9kID0gbG0oTU8yfkRPLCB3ZWlnaHQ9VywgZGF0YT1EYXRhW2xldGhhbCB8IEhpZ2hNTzIsXSkNCiMgVGhpcyBuZXcgcmVncmVzc2lvbiBpcyBhbHdheXMgYW4gaW1wcm92ZW1lbnQsIGJ1dCB0aGVyZSBjYW4gc3RpbGwNCiMgYmUgcG9pbnRzIGFib3ZlIHRoZSBsaW5lLCBzbyB3ZSByZXBlYXQNCnByZWRNTzJfMiA9IGFzLm51bWVyaWMocHJlZGljdCh0aGVNb2QsIGRhdGEuZnJhbWUoRE89RGF0YSRETykpKQ0KRGF0YSRkZWx0YTIgPSAoRGF0YSRNTzItcHJlZE1PMl8yKS9wcmVkTU8yXzIgKiAxMDANCkRhdGEkZGVsdGEyW0RhdGEkRE8gPCBwaXZvdERPXSA9IDANCnRvbCA9IERhdGEkZGVsdGEyW0hpZ2hNTzJdDQpIaWdoVmFsdWVzMiA9IERhdGEkZGVsdGEyID4gdG9sDQppZihzdW0oSGlnaFZhbHVlczIpPjApew0KUmFua3MyID0gcmFuaygtMSpEYXRhJGRlbHRhMikNCkhpZ2hNTzJfMiA9IEhpZ2hWYWx1ZXMyICYgUmFua3MyID09IDEgIyBrZWVwIHRoZSBsYXJnZXN0IHJlc2lkdWFsDQpuYmxldGhhbCA9IHN1bShsZXRoYWwpDQpEYXRhJFcgPSBOQQ0KRGF0YSRXW2xldGhhbF09MS9uYmxldGhhbA0KRGF0YSRXW0hpZ2hNTzJfMl0gPSAxDQp0aGVNb2QyID0gbG0oTU8yfkRPLCB3ZWlnaHQ9VywgZGF0YT1EYXRhW2xldGhhbCB8IEhpZ2hNTzJfMixdKQ0KIyBpcyBuZXcgc2xvcGUgc3RlZXBlciB0aGFuIHRoZSBvbGQgb25lPw0KaWYodGhlTW9kMiRjb2VmWzJdID4gdGhlTW9kJGNvZWZbMl0pIHsNCnRoZU1vZCA9IHRoZU1vZDINCkhpZ2hNTzIgPSBIaWdoTU8yXzINCn0NCn0gIyBlbmQgc2Vjb25kIHNlYXJjaCBmb3IgaGlnaCB2YWx1ZQ0KfSAjIGVuZCBmaXJzdCBzZWFyY2ggZm9yIGhpZ2ggdmFsdWUNCkNvZWYgPSBjb2VmZmljaWVudHModGhlTW9kKQ0KI1N0ZXAgNSwgY2hlY2sgZm9yIHBvc2l0aXZlIGludGVyY2VwdA0KQWJvdmVPcmlnaW4gPSBDb2VmWzFdID4gMA0KIyBpZiBpdCBpcywgd2UgdXNlIGEgcmVncmVzc2lvbiB0aGF0IGdvZXMgdGhyb3VnaCB0aGUgb3JpZ2luDQppZiAoQWJvdmVPcmlnaW4pew0KdGhlTW9kID0gbG0oTU8yfkRPIC0xLCBkYXRhPURhdGFbbGV0aGFsLF0pDQpDb2VmID0gYygwLCBjb2VmZmljaWVudHModGhlTW9kKSkgIyBuZWVkIHRvIGFkZCB0aGUgaW50ZXJjZXB0ICgwKQ0KIyBtYW51YWxseSB0byBoYXZlIGEgcGFpciBvZiBjb2VmZmljaWVudHMNCm1ldGhvZCA9ICJ0aHJvdWdoX29yaWdpbiINCkhpZ2hNTzIgPSByZXAoRkFMU0UsIG5yb3coRGF0YSkpICMgZGlkIG5vdCB1c2UgdGhlIGFkZGl0aW9uYWwgdmFsdWUNCiMgZnJvbSBTdGVwIDQNCn0NCnBvMmNyaXQgPSBhcy5udW1lcmljKHJvdW5kKChTTVIgLSBDb2VmWzFdKSAvIENvZWZbMl0sIDEpKQ0Kc3VtX21vZCA9IHN1bW1hcnkodGhlTW9kKQ0KYW5vdl9tb2QgPSBhbm92YSh0aGVNb2QpDQpPMkNSSVQgPSBsaXN0KG8yY3JpdD1wbzJjcml0LCBTTVI9U01SLCBOYl9NTzJfY29uZm9ybWluZyA9IE5fdW5kZXJfU01SLA0KTmJfTU8yX2NvbmZfdXNlZCA9IGZpbmFsX05fdW5kZXJfU01SLA0KSGlnaF9NTzJfcmVxdWlyZWQgPSBzdW0oSGlnaE1PMikgPT0gMSwgb3JpZ0RhdGE9RGF0YSwNCk1ldGhvZD1tZXRob2QsIG1vZD10aGVNb2QsIHIyID0gc3VtX21vZCRyLnNxdWFyZWQsDQpQID0gYW5vdl9tb2QkIlByKD5GKSIsIGxldGhhbFBvaW50cyA9IHdoaWNoKGxldGhhbCksDQpBZGRlZFBvaW50cyA9IHdoaWNoKEhpZ2hNTzIpKQ0KfSAjIGVuZCBmdW5jdGlvbg0KYGBgDQoNCmBwbG90TzJjcml0KClgOiAqKmF1dGhvcmVkIGJ5IERlbmlzIENoYWJvdCoqLCB1c2VkIHRvIHBsb3QgdGhlIG1vZGVzIHVzZWQgZm9yIHRoZSBgY2FsY08yY3JpdCgpYCBmdW5jdGlvbi4gQ2xhaXJlYXV4IGFuZCBDaGFib3QgKDIwMTYpIF5bMV1eICAgDQoNCioqKk5vdGU6IEkgYWRkZWQgYWJsaW5lKGg9bG93ZXN0TU8yLCBjb2w9InBpbmsiKSBzbyB0aGF0IEkgY291bGQgdmlzdWFsaXNlIHRoZSBsb3dlc3RNTzIgcG9zaXRpb24qKioNCg0KYGBge3J9DQpwbG90TzJjcml0IDwtIGZ1bmN0aW9uKG8yY3JpdG9iaiwgcGxvdElEPSIiLA0KWGxhYj0iRGlzc29sdmVkIG94eWdlbiAoJSBzYXQuKSIsIFlsYWI9ImRvdGl0YWx1bW9sIiwNCnNtci5jZXg9MC45LCBvMmNyaXQuY2V4PTAuOSwgcGxvdElELmNleD0xLjIsDQpUcmFuc3BhcmVuY3k9VCwuLi4pDQp7DQojIEFVVEhPUjogRGVuaXMgQ2hhYm90LCBJbnN0aXR1dCBNYXVyaWNlLUxhbW9udGFnbmUsIERGTywgQ2FuYWRhDQojIGZpcnN0IHZlcnNpb24gd3JpdHRlbiBpbiBKdW5lIDIwMDkNCiMgbGFzdCB1cGRhdGVkIDIwMTUtMDItMDkNCiMgZm9yIFIgcGxvdHRpbmcgZGV2aWNlcyB0aGF0IGRvIG5vdCBzdXBwb3J0IHRyYW5zcGFyZW5jeQ0KIyAoZS5nLiwgcG9zdHNjcmlwdCksIHNldCBUcmFuc3BhcmVuY3kgdG8gRkFMU0UNCnNtciA9IG8yY3JpdG9iaiRTTVINCmlmKFlsYWIgJWluJSBjKCJkb3RpdGFsdW1vbCIsICJpdGFsdW1vbCIsICJkb3R1bW9sIiwgInVtb2wiLA0KImRvdGl0YWxtZyIsICJpdGFsbWciLCAiZG90bWciLCAibWciKSkgew0Kc3dpdGNoKFlsYWIsDQpkb3RpdGFsdW1vbCA9IHsNCm1vMi5sYWIgPSBleHByZXNzaW9uKHBhc3RlKGl0YWxpYyhkb3QoTSkpW09bMl1dLCAiICgiLG11LCJtb2wgIiwgT1syXSwNCiIgIiwgbWluXi0xLCAiICIsIGtnXi0xLCAiKSIpKQ0KfSwNCml0YWx1bW9sID0gew0KbW8yLmxhYiA9IGV4cHJlc3Npb24ocGFzdGUoaXRhbGljKE0pW09bMl1dLCAiICgiLG11LCJtb2wgIiwgT1syXSwgIiAiLA0KbWluXi0xLCAiICIsIGtnXi0xLCAiKSIpKQ0KfSwNCmRvdHVtb2wgPSB7DQptbzIubGFiID0gZXhwcmVzc2lvbihwYXN0ZShkb3QoTSlbT1syXV0sICIgKCIsbXUsIm1vbCAiLCBPWzJdLCAiICIsDQptaW5eLTEsICIgIiwga2deLTEsICIpIikpDQp9LA0KdW1vbCA9IHsNCm1vMi5sYWIgPSBleHByZXNzaW9uKHBhc3RlKE1bT1syXV0sICIgKCIsbXUsIm1vbCAiLCBPWzJdLCAiICIsIG1pbl4tMSwNCiIgIiwga2deLTEsICIpIikpDQp9LA0KZG90aXRhbG1nID0gew0KbW8yLmxhYiA9IGV4cHJlc3Npb24ocGFzdGUoaXRhbGljKGRvdChNKSlbT1syXV0sICIgKG1nICIsIE9bMl0sICIgIiwNCmheLTEsICIgIiwga2deLTEsICIpIikpDQp9LA0KaXRhbG1nID0gew0KbW8yLmxhYiA9IGV4cHJlc3Npb24ocGFzdGUoaXRhbGljKE0pW09bMl1dLCAiIChtZyAiLCBPWzJdLCAiICIsDQpoXi0xLCAiICIsIGtnXi0xLCAiKSIpKQ0KfSwNCmRvdG1nID0gew0KbW8yLmxhYiA9IGV4cHJlc3Npb24ocGFzdGUoZG90KE0pW09bMl1dLCAiIChtZyAiLCBPWzJdLCAiICIsIGheLTEsICIgIiwNCmtnXi0xLCAiKSIpKQ0KfSwNCm1nID0gew0KbW8yLmxhYiA9IGV4cHJlc3Npb24ocGFzdGUoTVtPWzJdXSwgIiAobWcgIiwgT1syXSwgIiAiLCBoXi0xLCAiICIsDQprZ14tMSwgIikiKSkNCn0NCikNCn0gZWxzZSBtbzIubGFiPVlsYWINCmlmKFRyYW5zcGFyZW5jeSkge0NvbD1jKHJnYigwLDAsMCwwLjcpLCAicmVkIiwgIm9yYW5nZSIpDQp9IGVsc2Uge0NvbD1jKGdyZXkoMC4zKSwgInJlZCIsICJvcmFuZ2UiKX0NCkRhdGE9bzJjcml0b2JqJG9yaWdEYXRhDQpsb3dlc3RNTzIgPSBxdWFudGlsZShEYXRhJE1PMltEYXRhJERPID49IDgwXSwgcD0wLjA1KSAjIEkgYWRkZWQgdGhpcw0KRGF0YSRDb2xvciA9IENvbFsxXQ0KRGF0YSRDb2xvcltvMmNyaXRvYmokbGV0aGFsUG9pbnRzXSA9IENvbFsyXQ0KRGF0YSRDb2xvcltvMmNyaXRvYmokQWRkZWRQb2ludHNdID0gQ29sWzNdDQojIG9yZGluYXJ5IExTIHJlZ3Jlc3Npb24gd2l0aG91dCBhZGRlZCBwb2ludHM6IGJsdWUgbGluZSwgcmVkIHN5bWJvbHMNCiMgb3JkaW5hcnkgTFMgcmVncmVzc2lvbiB3aXRoIGFkZGVkIHBvaW50czogYmx1ZSBsaW5lLCByZWQgJiBvcmFuZ2Ugc3ltYm9scw0KIyByZWdyZXNzaW9uIHRocm91Z2ggb3JpZ2luOiBncmVlbiBkb3R0ZWQgbGluZSwgcmVkIHN5bWJvbHMNCmxpbmUuY29sb3IgPSBpZmVsc2UobzJjcml0b2JqJE1ldGhvZD09IkxTX3JlZyIsICJibHVlIiwgImRhcmtncmVlbiIpDQpsaW5lLnR5cGUgPSBpZmVsc2UobzJjcml0b2JqJE1ldGhvZD09IkxTX3JlZyIsIDEsIDMpDQpsaW1YID0gYygwLCBtYXgoRGF0YSRETykpDQpsaW1ZID0gYygwLCBtYXgoRGF0YSRNTzIpKQ0KcGxvdChNTzJ+RE8sIGRhdGE9RGF0YSwgeGxpbT1saW1YLCB5bGltPWxpbVksIGNvbD1EYXRhJENvbG9yLCB4bGFiPVhsYWIsDQp5bGFiPW1vMi5sYWIsIC4uLikNCmNvb3JkIDwtIHBhcigidXNyIikNCmlmKHBsb3RJRCAhPSAiIil7DQp0ZXh0KDAsIGNvb3JkWzRdLCBwbG90SUQsIGNleD1wbG90SUQuY2V4LCBhZGo9YygwLDEuMikpDQp9DQphYmxpbmUoaD1sb3dlc3RNTzIsIGNvbD0icGluayIpICMgSSBhZGRlZCB0aGlzDQphYmxpbmUoaD1zbXIsIGNvbD0ib3JhbmdlIikNCnRleHQoY29vcmRbMV0sIHNtciwgIlNNUiIsIGFkaj1jKC0wLjEsMS4zKSwgY2V4PXNtci5jZXgpDQp0ZXh0KGNvb3JkWzFdLCBzbXIsIHJvdW5kKHNtciwxKSwgYWRqPWMoLTAuMSwtMC4zKSwgY2V4PXNtci5jZXgpDQppZighaXMubmEobzJjcml0b2JqJG8yY3JpdCkpIHsNCmFibGluZShvMmNyaXRvYmokbW9kLCBjb2w9bGluZS5jb2xvciwgbHR5PWxpbmUudHlwZSkNCnNlZ21lbnRzKG8yY3JpdG9iaiRvMmNyaXQsIHNtciwgbzJjcml0b2JqJG8yY3JpdCwgY29vcmRbM10sDQpjb2w9bGluZS5jb2xvciwgbHdkPTEpDQp0ZXh0KHg9bzJjcml0b2JqJG8yY3JpdCwgeT0wLCBvMmNyaXRvYmokbzJjcml0LCBjb2w9bGluZS5jb2xvciwNCmNleD1vMmNyaXQuY2V4LCBhZGo9YygtMC4xLDAuNSkpDQp9DQp9ICMgZW5kIG9mIGZ1bmN0aW9uDQpgYGANCg0KDQo8IS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0+DQojIPCfk4IgRGlyZWN0b3JpZXMgDQo8IS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0+DQoNCiMjICBJbnB1dA0KDQoqKvCfk6UgaW5wdXRfZGF0YV93ZCoqOiBEaXJlY3RvcnkgZm9yIHRoZSBtZXRhZGF0YQ0KDQpgYGB7cn0NCndkIDwtIGdldHdkKCkNCmlucHV0X2RhdGFfd2QgPC0gcGFzdGUwKHdkLCAiLi9pbnB1dC1kYXRhIikgIyBjcmVhdGVzIGEgdmFyaWFibGUgd2l0aCB0aGUgbmFtZSBvZiB0aGUgd2Qgd2Ugd2FudCB0byB1c2UNCmBgYA0KDQoqKvCfk6UgbW9kX2RhdGFfd2QqKjogRGlyZWN0b3J5IGZvciBtb2RlbCBvdXRwdXQgZGF0YSBlc3RpbWF0ZWQgc2xvcGVzDQoNCmBgYHtyfQ0KbW9kX2RhdGFfd2QgPC0gcGFzdGUwKHdkLCAiLi9tb2QtZGF0YSIpDQpgYGANCg0KIyMgT3V0cHV0DQoNCioq8J+TpCBvdXRwdXRfZmlnX3dkKio6IHRoaXMgaXMgd2hlcmUgd2Ugd2lsbCBwdXQgdGhlIGZpZ3VyZXMNCg0KYGBge3J9DQpvdXRwdXRfZmlnX3dkIDwtIHBhc3RlMCh3ZCwgIi4vb3V0cHV0LWZpZyIpDQppZiAoIWRpci5leGlzdHMoIm91dHB1dC1maWciKSkge2Rpci5jcmVhdGUoIm91dHB1dC1maWciKX0NCmBgYA0KDQoqKvCfk6Qgb3V0cHV0X21vZHNfd2QqKjogdGhpcyBpcyB3aGVyZSB3ZSB3aWxsIHB1dCB0aGUgbW9kZWxzDQoNCmBgYHtyfQ0Kb3V0cHV0X21vZHNfd2QgPC0gcGFzdGUwKHdkLCAiLi9vdXRwdXQtbW9kIikNCmlmICghZGlyLmV4aXN0cygib3V0cHV0LW1vZCIpKSB7ZGlyLmNyZWF0ZSgib3V0cHV0LW1vZCIpfQ0KYGBgDQoNCjwhLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLT4NCiMg8J+SvyBEYXRhIA0KPCEtLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tPg0KDQojIyBTbG9wZXMgKE1P4oKCKQ0KDQoqKvCfkr8gc2xvcGVfZGYqKjogV2UgaGF2ZSBpbXBvcnRlZCB0aGUgc2xvcGVzIGV4dHJhY3RlZCBpbiBMYWJDaGFydCBkdXJpbmcgZWFjaCBwaGFzZSBvZiB0aGUgZXhwZXJpbWVudA0KDQpgYGB7cn0NCiBzZXR3ZChpbnB1dF9kYXRhX3dkKQ0KIyANCiMgIyBHZXQgdGhlIG5hbWVzIG9mIGFsbCBzaGVldHMgaW4gdGhlIEV4Y2VsIGZpbGUNCnNoZWV0X25hbWVzIDwtIGV4Y2VsX3NoZWV0cygibGFiY2hhcnQtYWxsLWRhdGVzLTIwMjUueGxzeCIpDQphbGxfdHJpYWxzX3NlbGVjdCA8LSBjKCJzdGFydF9kYXRlIiwgIm9yZGVyIiwgInBoYXNlIiwgImN5Y2xlIiwgImRhdGUiLCAidGltZSIpDQpzbG9wZV9saXN0IDwtIGxpc3QoKQ0KDQpmb3IgKHNoZWV0IGluIHNoZWV0X25hbWVzKSB7DQoNCiAgZGYgPC0gcmVhZF9leGNlbCgibGFiY2hhcnQtYWxsLWRhdGVzLTIwMjUueGxzeCIsIHNoZWV0ID0gc2hlZXQpICU+JSANCiAgZHBseXI6OnJlbmFtZV93aXRoKHRvbG93ZXIpDQogIA0KYV9uYW1lIDwtIHBhc3RlMCgiYV8iLCB0b2xvd2VyKHNoZWV0KSkNCmFfZGYgPC0gZGYgJT4lDQogIGRwbHlyOjpzZWxlY3Qoc3RhcnRzX3dpdGgoJ2EnKSwgYWxsX3RyaWFsc19zZWxlY3QpICU+JSANCiAgZHBseXI6OnJlbmFtZSh0ZW1wID0gYV90ZW1wKSAlPiUgDQogIGRwbHlyOjptdXRhdGUoYWNyb3NzKHN0YXJ0c193aXRoKCdhJyksIGFzLm51bWVyaWMpKSAlPiUgDQogIHBpdm90X2xvbmdlcigNCiAgICBjb2xzID0gc3RhcnRzX3dpdGgoJ2EnKSwgIyBTZWxlY3QgYWxsIGNvbHVtbnMgdG8gcGl2b3QNCiAgICBuYW1lc190byA9IGMoImNoYW1iZXJfaWQiLCAiLnZhbHVlIiksICMgU2VwYXJhdGUgY29sdW1uIG5hbWVzIGludG8gJ2lkJyBhbmQgb3RoZXIgdmFyaWFibGVzDQogICAgbmFtZXNfc2VwID0gIl8iDQogICkgJT4lDQogIGRwbHlyOjptdXRhdGUocmVzcGlyb21ldGVyX2dyb3VwID0gImEiKSAjIEFkZCBhIG5ldyBjb2x1bW4gd2l0aCBhIGZpeGVkIHZhbHVlDQoNCnNsb3BlX2xpc3RbW2FfbmFtZV1dPC0gYV9kZg0KDQpiX25hbWUgPC0gcGFzdGUwKCJiXyIsIHRvbG93ZXIoc2hlZXQpKQ0KYl9kZiA8LSBkZiAlPiUgDQogIGRwbHlyOjpzZWxlY3Qoc3RhcnRzX3dpdGgoJ2InKSwgYWxsX3RyaWFsc19zZWxlY3QpICU+JSANCiAgZHBseXI6OnJlbmFtZSh0ZW1wID0gYl90ZW1wKSAlPiUgDQogIGRwbHlyOjptdXRhdGUoYWNyb3NzKHN0YXJ0c193aXRoKCdiJyksIGFzLm51bWVyaWMpKSAlPiUgDQogIHBpdm90X2xvbmdlcigNCiAgICBjb2xzID0gc3RhcnRzX3dpdGgoJ2InKSwgIyBTZWxlY3QgYWxsIGNvbHVtbnMgdG8gcGl2b3QNCiAgICBuYW1lc190byA9IGMoImNoYW1iZXJfaWQiLCAiLnZhbHVlIiksICMgU2VwYXJhdGUgY29sdW1uIG5hbWVzIGludG8gJ2lkJyBhbmQgb3RoZXIgdmFyaWFibGVzDQogICAgbmFtZXNfc2VwID0gIl8iDQogICkgJT4lIA0KICAgIGRwbHlyOjptdXRhdGUocmVzcGlyb21ldGVyX2dyb3VwID0gImIiKQ0KDQpzbG9wZV9saXN0W1tiX25hbWVdXSA8LSBiX2RmDQoNCmNfbmFtZSA8LSBwYXN0ZTAoImNfIiwgdG9sb3dlcihzaGVldCkpDQpjX2RmIDwtIGRmICU+JSANCiAgZHBseXI6OnNlbGVjdChzdGFydHNfd2l0aCgnYycpLCBhbGxfdHJpYWxzX3NlbGVjdCkgJT4lIA0KICBkcGx5cjo6cmVuYW1lKHRlbXAgPSBjX3RlbXAsDQogICAgICAgICAgICAgICAgaV9jeWNsZSA9IGN5Y2xlKSAlPiUgDQogIGRwbHlyOjptdXRhdGUoYWNyb3NzKHN0YXJ0c193aXRoKCdjJyksIGFzLm51bWVyaWMpKSAlPiUNCiAgcGl2b3RfbG9uZ2VyKA0KICAgIGNvbHMgPSBzdGFydHNfd2l0aCgnYycpLCAjIFNlbGVjdCBhbGwgY29sdW1ucyB0byBwaXZvdA0KICAgIG5hbWVzX3RvID0gYygiY2hhbWJlcl9pZCIsICIudmFsdWUiKSwgIyBTZXBhcmF0ZSBjb2x1bW4gbmFtZXMgaW50byAnaWQnIGFuZCBvdGhlciB2YXJpYWJsZXMNCiAgICBuYW1lc19zZXAgPSAiXyINCiAgKSAlPiUgDQogICAgZHBseXI6Om11dGF0ZShyZXNwaXJvbWV0ZXJfZ3JvdXAgPSAiYyIpICU+JSANCiAgZHBseXI6OnJlbmFtZShjeWNsZSA9IGlfY3ljbGUpDQoNCnNsb3BlX2xpc3RbW2NfbmFtZV1dIDwtIGNfZGYNCg0KZF9uYW1lIDwtIHBhc3RlMCgiZF8iLCB0b2xvd2VyKHNoZWV0KSkNCmRfZGYgPC0gZGYgJT4lIA0KICBkcGx5cjo6c2VsZWN0KHN0YXJ0c193aXRoKCdkJyksIGFsbF90cmlhbHNfc2VsZWN0KSAlPiUgDQogIGRwbHlyOjpyZW5hbWUodGVtcCA9IGRfdGVtcCwNCiAgICAgICAgICAgICAgICBpX2RhdGUgPSBkYXRlKSAlPiUgDQogIGRwbHlyOjptdXRhdGUoYWNyb3NzKHN0YXJ0c193aXRoKCdkJyksIGFzLm51bWVyaWMpKSAlPiUNCiAgcGl2b3RfbG9uZ2VyKA0KICAgIGNvbHMgPSBzdGFydHNfd2l0aCgnZCcpLCAjIFNlbGVjdCBhbGwgY29sdW1ucyB0byBwaXZvdA0KICAgIG5hbWVzX3RvID0gYygiY2hhbWJlcl9pZCIsICIudmFsdWUiKSwgIyBTZXBhcmF0ZSBjb2x1bW4gbmFtZXMgaW50byAnaWQnIGFuZCBvdGhlciB2YXJpYWJsZXMNCiAgICBuYW1lc19zZXAgPSAiXyINCiAgKSAlPiUgDQogICAgZHBseXI6Om11dGF0ZShyZXNwaXJvbWV0ZXJfZ3JvdXAgPSAiZCIpICU+JSANCiAgZHBseXI6OnJlbmFtZShkYXRlID0gaV9kYXRlKQ0KDQpzbG9wZV9saXN0W1tkX25hbWVdXSA8LSBkX2RmDQp9DQoNCg0Kc2xvcGVfZGYgPC0gYmluZF9yb3dzKHNsb3BlX2xpc3QpICU+JSANCiAgZHBseXI6Om11dGF0ZShyZXNwX2NhdF9kYXRlID0gcGFzdGUwKHJlc3Bpcm9tZXRlcl9ncm91cCwgIl8iLCBzdGFydF9kYXRlKSwNCiAgICAgICAgICAgICAgICBjaGFtYmVyX24gPSBzdHJfZXh0cmFjdChjaGFtYmVyX2lkLCAiXFxkKyIpLA0KICAgICAgICAgICAgICAgIGlkX3Byb3ggPSBwYXN0ZTAocmVzcF9jYXRfZGF0ZSwgIl8iLCBjaGFtYmVyX24pLA0KICAgICAgICAgICAgICAgIHRpbWVfaG1zID0gYXNfaG1zKHRpbWUqMzYwMCksDQogICAgICAgICAgICAgICAgZGF0ZV9jaHIgPSBmb3JtYXQoZGF0ZSwgIiVkLyVtLyVZIikNCiAgICAgICAgICAgICAgICApDQpgYGANCg0KIyMgRmlzaCBtZXRhZGF0YQ0KDQoqKvCfkr8gbWV0YWRhdGEqKjogVGhpcyBpcyB0aGUgbWV0YSBkYXRhIGZvciBlYWNoIGNoYW1iZXIgICANCg0KKk5vdGU6IFdlIGFyZSBhbHNvIGFkZGluZyB2b2x1bWUgYmFzZWQgb24gY2hhbWJlciB0eXBlLiogICANCg0KYGBge3J9DQpzZXR3ZChpbnB1dF9kYXRhX3dkKQ0KDQptZXRhZGF0YSA8LSByZWFkX2V4Y2VsKCJmaXNoLW1ldGEueGxzeCIsIG5hID0gIk5BIikgJT4lIA0KICBkcGx5cjo6bXV0YXRlKGlkX3NwbGl0ID0gaWQpICU+JSANCiAgdGlkeXI6OnNlcGFyYXRlKGlkX3NwbGl0LCBpbnRvID0gYygicmVzcGlyb21ldGVyX2dyb3VwIiwgInNhbGluaXR5X2dyb3VwIiwgInN0YXJ0X2RhdGUiLCAiY2hhbWJlciIpLCBzZXAgPSAiXyIpICU+JSANCiAgZHBseXI6Om11dGF0ZSgNCiAgICAgIHZvbHVtZSA9IGRwbHlyOjpjYXNlX3doZW4oDQogICAgICAgIGNoYW1iZXJfdHlwZSA9PSAiTCIgfiAwLjMwMCwNCiAgICAgICAgY2hhbWJlcl90eXBlID09ICJNX00iIH4gMC4xMDUsDQogICAgICAgIGNoYW1iZXJfdHlwZSA9PSAiTV9OTSIgfiAwLjExLA0KICAgICAgICBjaGFtYmVyX3R5cGUgPT0gIlMiIH4gMC4wNTgsDQogICAgICAgIGNoYW1iZXJfdHlwZSA9PSAiU00iIH4gMC4wNzUsDQogICAgICAgIGNoYW1iZXJfdHlwZSA9PSAiRDMiIH4gMC4wNTUsDQogICAgICAgIFRSVUUgfiBOQQ0KICAgICAgKSwNCiAgICAgIGlkX3Byb3ggPSBwYXN0ZTAocmVzcGlyb21ldGVyX2dyb3VwLCAiXyIsIHN0YXJ0X2RhdGUsICJfIiwgY2hhbWJlciksDQogICAgICByZWxfc2l6ZSA9IG1hc3Mvdm9sdW1lKQ0KYGBgDQoNCg0KIyMgVXJiaW5hLCBHbG92ZXIsIGFuZCBGb3JzdGVyICgyMDEyKQ0KDQoqKvCfkr8gdXJiaW5hX2V0X2FsXzIwMTIqKjogVGhpcyBpcyB0aGUgbWVhbiBsZXZlbCBkYXRhIGV4dHJhY3RlZCBmcm9tIFVyYmluYSwgR2xvdmVyLCBhbmQgRm9yc3RlciAoMjAxMileWzJdXiBGaWd1cmUgMWEuIFdlIHVzZWQgdGhlIG1ldGFEaWdpdGlzZSBwYWNrYWdlIGluIFIgdG8gZXh0cmFjdCB0aGUgZGF0YSBeWzNdXi4gICANCg0KYGBge3J9DQpzZXR3ZChpbnB1dF9kYXRhX3dkKQ0KDQp1cmJpbmFfZXRfYWxfMjAxMiA8LSByZWFkLmNzdigidXJiaW5hLWV0LWFsLTIwMTItZmlnMWEuY3N2IikNCmBgYA0KDQojIyBQY3JpdCB2aXVzYWwgaW5zcGVjdGlvbg0KKirwn5K/IHBjcml0X2NoZWNrKio6IFRoaXMgZGF0YSBmcmFtZSBpcyBhIGxpc3Qgb2YgYWxsIGZpc2ggdGhhdCBoYWQgaGlnaGVyIG9yZGVyIHBvbHlub21pYWxzICgyXm5kXiBhbmQgM15yZF4gb3JkZXI7IHNlZSAnSW5jcmVtZW50YWwgcmVncmVzc2lvbiBhbmFseXNlcycgZm9yIGRldGFpbHMpIGFuZCBzdWJzZXF1ZW50bHkgaGFkIFBjcml0IG1vZGVsbGVkIChzZWUgJ1Bjcml0IG1vZGVsJyBmb3IgZGV0YWlscykuIEluIHRoaXMgZGF0YSB3ZSBoYXZlIGNoZWNrZWQgZWFjaCBQY3JpdCBtb2RlbCBhbmQgY29uZmlybWVkIHZpc3VhbCBpZiBhIFBjcml0IGlzIHByZXNlbnQuIA0KDQpgYGB7cn0NCnNldHdkKGlucHV0X2RhdGFfd2QpDQoNCnBjcml0X2NoZWNrIDwtIHJlYWRfZXhjZWwoInBjcml0LWNoZWNrLnhsc3giLCBuYSA9ICJOQSIpDQpgYGANCg0KDQojIyMgQ29tYmluZGluZyBtZXRhZGF0YQ0KDQpBZGRpbmcgdGhlIG1ldGEgZGF0YSB0byB0aGUgc2xvcGVzIGRhdGEgZnJhbWUgICAgICANCg0KYGBge3J9DQpzbG9wZV9kZl8yIDwtIHNsb3BlX2RmICU+JSANCiAgZHBseXI6OnNlbGVjdCgtc3RhcnRfZGF0ZSwgLXJlc3Bpcm9tZXRlcl9ncm91cCkgJT4lIA0KICBsZWZ0X2pvaW4obWV0YWRhdGEsIGJ5ID0gImlkX3Byb3giKSAlPiUgDQogIGRwbHlyOjptdXRhdGUobGlnaHRfZGFyayA9IGlmX2Vsc2UodGltZV9obXMgPj0gYXMuaG1zKCIwNzowMDowMCIpICYgdGltZV9obXMgPCBhcy5obXMoIjE5OjAwOjAwIiksICJsaWdodCIsICJkYXJrIikpICU+JSANCiAgZHBseXI6OmFycmFuZ2UoaWQpDQpgYGANCg0KPCEtLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tPg0KIyDwn6e8IERhdGEgdGlkeQ0KPCEtLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tPg0KDQojIyBOdW1iZXJzIA0KDQpXZSBoYXZlICoqNjQgZmlzaCoqIHdpdGggTU/igoIgZGF0YSAgIA0KDQpgYGB7cn0NCm4gPC0gc2xvcGVfZGZfMiAlPiUgDQogIGRwbHlyOjpmaWx0ZXIoY2hhbWJlcl9jb25kaXRpb24gPT0gImZpc2giKSAlPiUgDQogIGRwbHlyOjpkaXN0aW5jdChpZCkgJT4lIA0KICBucm93KC4pDQoNCnBhc3RlMCgibiA9ICIsIG4pDQpgYGANCg0KV2l0aCA0OCBmcm9tIHRoZSAwIHBwdCBhbmQgNDggZnJvbSA5IHBwdCBncm91cHMgICANCg0KYGBge3J9DQpzbG9wZV9kZl8yICU+JSANCiAgZHBseXI6OmZpbHRlcihjaGFtYmVyX2NvbmRpdGlvbiA9PSAiZmlzaCIpICU+JQ0KICBkcGx5cjo6Z3JvdXBfYnkoc2FsaW5pdHlfZ3JvdXApICU+JSANCiAgZHBseXI6OnJlZnJhbWUoJ24gdG90YWwnID0gbGVuZ3RoKHVuaXF1ZShpZCkpKSAlPiUgDQogIGd0KCkgJT4lIA0KICBjb2xzX2xhYmVsKA0KICAgIHNhbGluaXR5X2dyb3VwID0gIlNhbGluaXR5IGdyb3VwIg0KICApICU+JSANCiAgY29sc19hbGlnbigNCiAgICBhbGlnbiA9ICJjZW50ZXIiLCANCiAgICBjb2x1bW5zID0gZXZlcnl0aGluZygpDQogICkNCmBgYA0KDQojIyBGaXNoIHNpemUNCg0KSGVyZSB3ZSBjYWxjdWxhdGUgdGhlIG1lYW4gbGVuZ3RoIGFuZCBzaXplIG9mIGZpc2ggdXNlZCBpbiB0aGUgZXhwZXJpbWVudC4gICANCmBgYHtyfQ0KbWFzc19sZW5ndGggPC0gc2xvcGVfZGZfMiAlPiUgDQogIGRwbHlyOjpmaWx0ZXIoY2hhbWJlcl9jb25kaXRpb24gPT0gImZpc2giKSAlPiUNCiAgZHBseXI6Omdyb3VwX2J5KGlkKSAlPiUgDQogIGRwbHlyOjpzYW1wbGVfbigxKSAlPiUgDQogIGRwbHlyOjp1bmdyb3VwKCkgJT4lIA0KICBkcGx5cjo6cmVmcmFtZSh4X21hc3MgPSByb3VuZChtZWFuKG1hc3MsIG5hLnJtID0gVFJVRSksIDMpLA0KICAgICAgICAgICAgICAgICBtaW5fbWFzcyA9IHJvdW5kKG1pbihtYXNzLCBuYS5ybSA9IFRSVUUpLCAzKSwNCiAgICAgICAgICAgICAgICAgbWF4X21hc3MgPSByb3VuZChtYXgobWFzcywgbmEucm0gPSBUUlVFKSwgMyksDQogICAgICAgICAgICAgICAgIHhfbGVuZ3RoID0gcm91bmQobWVhbihsZW5ndGgsIG5hLnJtID0gVFJVRSksIDIpLA0KICAgICAgICAgICAgICAgICBtaW5fbGVuZ3RoID0gcm91bmQobWluKGxlbmd0aCwgbmEucm0gPSBUUlVFKSwgMiksDQogICAgICAgICAgICAgICAgIG1heF9sZW5ndGggPSByb3VuZChtYXgobGVuZ3RoLCBuYS5ybSA9IFRSVUUpLCAyKSkNCg0KbWFzc19tZWFuIDwtIG1hc3NfbGVuZ3RoICU+JSANCiAgcHVsbCh4X21hc3MpDQoNCm1hc3NfbWluIDwtIG1hc3NfbGVuZ3RoICU+JSANCiAgcHVsbChtaW5fbWFzcykNCg0KbWFzc19tYXggPC0gbWFzc19sZW5ndGggJT4lIA0KICBwdWxsKG1heF9tYXNzKQ0KDQpsZW5ndGhfbWVhbiA8LSBtYXNzX2xlbmd0aCAlPiUgDQogIHB1bGwoeF9sZW5ndGgpDQoNCmxlbmd0aF9taW4gPC0gbWFzc19sZW5ndGggJT4lIA0KICBwdWxsKG1pbl9sZW5ndGgpDQoNCmxlbmd0aF9tYXggPC0gbWFzc19sZW5ndGggJT4lIA0KICBwdWxsKG1heF9sZW5ndGgpDQoNCnBhc3RlMCgiVGhlIG1lYW4gbWFzcyBvZiBmaXNoIHdhcyAiLCBtYXNzX21lYW4sICIgZyAocmFuZ2U6ICIsIG1hc3NfbWluLCAiLSIsIG1hc3NfbWF4LCAiKSIsDQogICAgICAgIiwgYW5kIHRoZSBtZWFuIGxlbmd0aCB3YXMgIiwgbGVuZ3RoX21lYW4sICIgbW0gKHJhbmdlOiAiLCBsZW5ndGhfbWluLCAiLSIsIGxlbmd0aF9tYXgsICIpIikNCmBgYA0KDQoNCiMjIEZpbHRlcmluZyB0cmlhbHMNCg0KV2Ugd2lsbCByZW1vdmUgNiB0cmlhbHMgd2hpY2ggaGFkIGVycm9ycy4gVGhlc2UgYXJlIGFzIGZvbGxvd3M6ICAgDQoNCi0JYV8wXzI1bm92XzMgZmlzaCBkaWVkIGR1cmluZyB0cmlhbA0KLQliXzBfMjZub3ZfNCBmbGF0IGxpbmVkIGVhcmx5DQotCWNfMF8yMm5vdl8yIGNoYW1iZXIgd2FzIG9wZW5lZCBlYXJseQ0KLQljXzlfMjZub3ZfMiBjaGFtYmVyIHdhcyBvcGVuZWQgZWFybHkNCi0JY185XzI2bm92XzQgY2hhbWJlciB3YXMgb3BlbmVkIGVhcmx5DQotCWRfOV8yN25vdl8zIHNlbnNvciB3YXMganVtcHkgYW5kIGVuZCBwb2ludHMgd2VyZSBoYXJkIHRvIGNvbmZpZGVudGx5IElEIHZpc3VhbGx5ICAgDQoNCg0KYGBge3J9DQpyZW1vdmVfdHJpYWxfZXJyb3IgPC0gYygiYV8wXzI1bm92XzMiLCAiYl8wXzI2bm92XzQiLCAiY18wXzIybm92XzIiLCAiY185XzI2bm92XzIiLCAiY185XzI2bm92XzQiLCAiZF85XzI3bm92XzMiKQ0KDQpzbG9wZV9kZl9maWx0ZXIgPC0gc2xvcGVfZGZfMiAlPiUgDQogIGRwbHlyOjpmaWx0ZXIoIShpZCAlaW4lIHJlbW92ZV90cmlhbF9lcnJvcikpDQpgYGANCg0KV2Ugbm93IGhhdmUgKio1OCBmaXNoKiogd2l0aCBNTzIgZGF0YSAgIA0KDQpgYGB7cn0NCm4gPC0gc2xvcGVfZGZfZmlsdGVyICU+JSANCiAgZHBseXI6OmZpbHRlcihjaGFtYmVyX2NvbmRpdGlvbiA9PSAiZmlzaCIpICU+JSANCiAgZHBseXI6OmRpc3RpbmN0KGlkKSAlPiUgDQogIG5yb3coLikNCg0KcGFzdGUwKCJuID0gIiwgbikNCmBgYA0KDQpXaXRoIDMwIGluIHRoZSAwIHBwdCBncm91cCBhbmQgMjggaW4gdGhlIDkgcHB0IGdyb3VwICAgDQoNCmBgYHtyfQ0Kc2xvcGVfZGZfZmlsdGVyICU+JSANCiAgZHBseXI6OmZpbHRlcihjaGFtYmVyX2NvbmRpdGlvbiA9PSAiZmlzaCIpICU+JQ0KICBkcGx5cjo6Z3JvdXBfYnkoc2FsaW5pdHlfZ3JvdXApICU+JSANCiAgZHBseXI6OnJlZnJhbWUoJ24gdG90YWwnID0gbGVuZ3RoKHVuaXF1ZShpZCkpKSAlPiUgDQogIGd0KCkgJT4lIA0KICBjb2xzX2xhYmVsKA0KICAgIHNhbGluaXR5X2dyb3VwID0gIlNhbGluaXR5IGdyb3VwIg0KICApICU+JSANCiAgY29sc19hbGlnbigNCiAgICBhbGlnbiA9ICJjZW50ZXIiLCANCiAgICBjb2x1bW5zID0gZXZlcnl0aGluZygpDQogICkNCmBgYA0KDQojIyBGaWx0ZXJpbmcgTU/igoIgZXN0aW1hdGVzDQoNCkhlcmUgd2UgYXBwbHkgdGhlIGZvbGxvd2luZyBmaWx0ZXJzIHRvIHRoZSBNT+KCgiBkYXRhOiAgDQoNCi0gUmVtb3ZlIHRoZSBmaXJzdCA1IFNNUiBjeWNsZXMgKGJ1cm4gaW4pDQotIFJlbW92ZSBhbGwgcG9zaXRpdmUgcmF3IHNsb3Blcw0KLSBSZW1vdmUgYWxsIE1P4oKCIGNhbGN1bGF0ZWQgdXNpbmcgbGVzcyB0aGVuIDYwIGRhdGEgcG9pbnRzICg1IG1pbikgDQotIFJlbW92ZSBhbGwgTU/igoIgY2FsY3VsYXRlZCBpZiBP4oKCIGluY3JlYXNlcyBpbiBhIGNsb3NlZCBwaGFzZSAoaS5lLiB0cmlhbCBoYXMgZW5kZWQpICANCg0KDQpDaGVjayBwb3NpdGl2ZSB2YWx1ZXMgZm9yIE1P4oKCIGJlZm9yZSByZW1vdmluZy4gIA0KDQpgYGB7cn0NCnNsb3BlX3RpZHlfcmVtb3ZlX2ZsdXNoIDwtIHNsb3BlX2RmX2ZpbHRlciAlPiUNCiAgZHBseXI6OmZpbHRlcihwaGFzZSAhPSAic21yIiwgY2hhbWJlcl9jb25kaXRpb24gPT0gImZpc2giKSAlPiUgDQogIGRwbHlyOjpncm91cF9ieShpZCkgJT4lDQogIGRwbHlyOjphcnJhbmdlKGlkLCBvcmRlcikgJT4lICAjIEVuc3VyZSB0aGUgZGF0YSBpcyBzb3J0ZWQgd2l0aGluIGVhY2ggZ3JvdXANCiAgZHBseXI6Om11dGF0ZShvMl9kaWZmID0gaWZfZWxzZShyb3dfbnVtYmVyKCkgPT0gMSwgMCwgbzIgLSBsYWcobzIpKSwgIyBDYWxjdWxhdGUgdGhlIGRpZmZlcmVuY2UgaW4gJ28yJw0KICAgICAgICAgICAgICAgIG8yX2RpZmZfY3Vtc3VtID0gY3Vtc3VtKG8yX2RpZmYgPiAxKSkgJT4lICAjIENoZWNrcyBmaXJzdCBvY2N1cnJlbmNlIGFuZCBzdW1zIA0KICBkcGx5cjo6ZmlsdGVyKG8yX2RpZmZfY3Vtc3VtID09IDApICU+JSAgIyBLZWVwIHJvd3MgdW50aWwgdGhlIGZpcnN0IGp1bXAgPiAxDQogIGRwbHlyOjp1bmdyb3VwKCkgJT4lDQogIGRwbHlyOjpzZWxlY3QoLW8yX2RpZmYsIC1vMl9kaWZmX2N1bXN1bSkNCg0KcG9zdGl2ZV9zbG9wZXMgPC0gc2xvcGVfdGlkeV9yZW1vdmVfZmx1c2ggJT4lDQogIGRwbHlyOjpmaWx0ZXIoY2hhbWJlcl9jb25kaXRpb24gPT0gImZpc2giICYgbW8yY29yciA+IDApDQoNCmxpc3RfcG9zdGl2ZV9hbGwgPC0gc2xvcGVfZGZfZmlsdGVyICU+JSANCiAgZHBseXI6OmZpbHRlcihjaGFtYmVyX2NvbmRpdGlvbiA9PSAiZmlzaCIgJiBtbzJjb3JyID4gMCAmIHBoYXNlID09ICJzbXIiKSAlPiUgDQogIGRwbHlyOjpiaW5kX3Jvd3MoLiwgcG9zdGl2ZV9zbG9wZXMpICU+JSANCiAgZHBseXI6OmRpc3RpbmN0KGlkKSAlPiUgDQogIGRwbHlyOjpwdWxsKGlkKQ0KDQpwcmludChwYXN0ZTAoIlRoZXJlIGFyZSAiLCBsZW5ndGgobGlzdF9wb3N0aXZlX2FsbCksICIgZmlzaCB3aXRoIHBvc3RpdmUgc2xvcGVzLiBUaGVzZSBmaXNoIGFyZTogIiwgcGFzdGUobGlzdF9wb3N0aXZlX2FsbCwgY29sbGFwc2UgPSAiLCAiKSwgIi4iKSkNCmBgYA0KDQpGaWx0ZXJpbmcgdGhlIE1P4oKCIGRhdGEgICANCg0KYGBge3J9DQpjeWNsZV9idXJuIDwtIDA6NA0KDQpzbG9wZV9kZl9maWx0ZXJfMSA8LSBzbG9wZV9kZl9maWx0ZXIgJT4lDQogIGRwbHlyOjpmaWx0ZXIoIShjeWNsZSAlaW4lIGN5Y2xlX2J1cm4pICYgDQogICAgICAgICAgICAgICAgICBtbzJjb3JyIDwgMCAmIA0KICAgICAgICAgICAgICAgICAgbiA+IDYwICYNCiAgICAgICAgICAgICAgICAgIGNoYW1iZXJfY29uZGl0aW9uID09ICJmaXNoIg0KICAgICAgICAgICAgICAgICkNCiAgDQojIEZpbHRlciBvdXQgdGhlIGVuZCBmbHVzaA0Kc2xvcGVfdGlkeV9jbG9zZWQgPC0gc2xvcGVfZGZfZmlsdGVyXzEgJT4lDQogIGRwbHlyOjpmaWx0ZXIocGhhc2UgIT0gInNtciIpICU+JSANCiAgZHBseXI6Omdyb3VwX2J5KGlkKSAlPiUNCiAgZHBseXI6OmFycmFuZ2UoaWQsIG9yZGVyKSAlPiUgICMgRW5zdXJlIHRoZSBkYXRhIGlzIHNvcnRlZCB3aXRoaW4gZWFjaCBncm91cA0KICBkcGx5cjo6bXV0YXRlKG8yX2RpZmYgPSBpZl9lbHNlKHJvd19udW1iZXIoKSA9PSAxLCAwLCBvMiAtIGxhZyhvMikpLCAjIENhbGN1bGF0ZSB0aGUgZGlmZmVyZW5jZSBpbiAnbzInDQogICAgICAgICAgICAgICAgbzJfZGlmZl9jdW1zdW0gPSBjdW1zdW0obzJfZGlmZiA+IDEpKSAlPiUgICMgQ2hlY2tzIGZpcnN0IG9jY3VycmVuY2UgYW5kIHN1bXMgDQogIGRwbHlyOjpmaWx0ZXIobzJfZGlmZl9jdW1zdW0gPT0gMCkgJT4lICAjIEtlZXAgcm93cyB1bnRpbCB0aGUgZmlyc3QganVtcCA+IDENCiAgZHBseXI6OnVuZ3JvdXAoKSAlPiUNCiAgZHBseXI6OnNlbGVjdCgtbzJfZGlmZiwgLW8yX2RpZmZfY3Vtc3VtKQ0KDQpzbG9wZV90aWR5X3NtciA8LSBzbG9wZV9kZl9maWx0ZXJfMSAlPiUgDQogIGRwbHlyOjpmaWx0ZXIocGhhc2UgPT0gInNtciIpDQoNCnNsb3BlX2RmX2ZpbHRlcl8yIDwtIHJiaW5kKHNsb3BlX3RpZHlfc21yLCBzbG9wZV90aWR5X2Nsb3NlZCkgJT4lIA0KICBkcGx5cjo6YXJyYW5nZShpZCwgb3JkZXIpDQpgYGANCg0KIyMgQ2FsY3VsYXRpbmcgU01SDQoNCldlIHdpbGwgY2FsY3VsYXRlIFNNUiB1c2luZyBgY2FsY1NNUmAgZnVuY3Rpb24gYnkgQ2hhYm90LCBTdGVmZmVuc2VuIGFuZCBGYXJyZWxsICgyMDE2KV5bMV1eLiBTcGVjaWZpY2FsbHksIHdlIHVzZSBtZWFuIG9mIHRoZSBsb3dlc3Qgbm9ybWFsIGRpc3RyaWJ1dGlvbiAoTUxORCkgd2hlcmUgQ1ZtbG5kIDwgNS40IGFuZCB0aGUgbWVhbiBvZiB0aGUgbG93ZXIgMjAlIHF1YW50aWxlIChxMC4yKSB3ZXJlIENWbWxuZCA+IDUuNC4gSWYgQ1ZtbG5kIGlzIG5vdCBjYWxjdWxhdGVkIHdlIGhhdmUgdXNlZCBxMC4yLiAgDQoNCmBgYHtyfQ0KbGFiY2hhcnRfY2hhYm90X3NtciA8LSBzbG9wZV9kZl9maWx0ZXJfMiAlPiUNCiAgZHBseXI6OmZpbHRlcihwaGFzZSA9PSAic21yIikNCg0KIyBFeHRyYWN0IGRpc3RpbmN0IElEcw0KaWRzIDwtIGxhYmNoYXJ0X2NoYWJvdF9zbXIgJT4lIA0KICBkcGx5cjo6ZGlzdGluY3QoaWQpICU+JSANCiAgZHBseXI6OnB1bGwoKQ0KDQojIEluaXRpYWxpc2UgYW4gZW1wdHkgbGlzdCB0byBzdG9yZSBTTVIgZGF0YQ0Kc21yX2xpc3QgPC0gbGlzdCgpDQoNCiMgUHJvY2VzcyBlYWNoIElEDQpmb3IgKGlkX2kgaW4gaWRzKSB7DQogIHRyeUNhdGNoKHsNCiAgICAjIEZpbHRlciB0aGUgZGF0YSBmb3IgdGhlIGN1cnJlbnQgSUQNCiAgICBkZl9pIDwtIGxhYmNoYXJ0X2NoYWJvdF9zbXIgJT4lIA0KICAgICAgZHBseXI6OmZpbHRlcihpZCA9PSBpZF9pKSAlPiUgDQogICAgICBkcGx5cjo6bXV0YXRlKGFic19tbzJjb3JyID0gYWJzKG1vMmNvcnIpKQ0KICAgIA0KICAgICMgQ2FsY3VsYXRlIFNNUiByZXN1bHRzDQogICAgY2FsY1NNUl9yZXN1bHRzIDwtIGNhbGNTTVIoZGZfaSRhYnNfbW8yY29ycikNCiAgICBDVm1sbmRfaSA8LSBjYWxjU01SX3Jlc3VsdHMkQ1ZtbG5kDQogICAgcXVhbnRfaSA8LSBjYWxjU01SX3Jlc3VsdHMkcXVhbnQgJT4lIGFzX3RpYmJsZSgpDQogICAgcXVhbnRfMjBwZXJfaSA8LSBxdWFudF9pJHZhbHVlWzNdDQogICAgbWxuZF9pIDwtIGNhbGNTTVJfcmVzdWx0cyRtbG5kDQogICAgc21yX3ZhbHVlIDwtIGlmX2Vsc2UoQ1ZtbG5kX2kgPCA1LjQsIG1sbmRfaSwgcXVhbnRfMjBwZXJfaSkNCiAgICBzbXJfdHlwZSA8LSBpZl9lbHNlKENWbWxuZF9pIDwgNS40LCAibWxuZCIsICJxdWFudF8yMHBlciIpDQogICAgc21yX3ZhbHVlIDwtIGlmX2Vsc2UoaXMubmEoc21yX3ZhbHVlKSwgcXVhbnRfMjBwZXJfaSwgc21yX3ZhbHVlKQ0KICAgIHNtcl90eXBlIDwtIGlmX2Vsc2UoaXMubmEoc21yX3R5cGUpLCAicXVhbnRfMjBwZXIiLCBzbXJfdHlwZSkNCiAgICANCiAgICAjIENyZWF0ZSBhIGRhdGEgZnJhbWUgZm9yIHRoZSBjdXJyZW50IElEDQogICAgc21yX2RmIDwtIHRpYmJsZSgNCiAgICAgIGlkID0gaWRfaSwNCiAgICAgIHNtciA9IHNtcl92YWx1ZSwNCiAgICAgIHNtcl9lc3QgPSBzbXJfdHlwZQ0KICAgICkNCiAgICANCiAgfSwgZXJyb3IgPSBmdW5jdGlvbihlKSB7DQogICAgIyBIYW5kbGUgZXJyb3JzIGJ5IGFzc2lnbmluZyBOQSB2YWx1ZXMNCiAgICBzbXJfZGYgPC0gdGliYmxlKA0KICAgICAgaWQgPSBpZF9pLA0KICAgICAgc21yID0gTkEsDQogICAgICBzbXJfZXN0ID0gTkENCiAgICApDQogIH0pDQogIA0KICAjIEFwcGVuZCB0byB0aGUgbGlzdA0KICBzbXJfbGlzdFtbaWRfaV1dIDwtIHNtcl9kZg0KfQ0KDQojIENvbWJpbmUgYWxsIGluZGl2aWR1YWwgU01SIGRhdGEgZnJhbWVzIGludG8gb25lDQpzbXJfZGYgPC0gYmluZF9yb3dzKHNtcl9saXN0KSAlPiUgDQogIGRwbHlyOjpyZW5hbWUoc21yX2NoYWJvdCA9IHNtciwNCiAgICAgICAgICAgICAgICBzbXJfY2hhYm90X21ldGhvZCA9IHNtcl9lc3QpDQoNCnNsb3BlX2RmX2ZpbHRlcl8zIDwtIHNsb3BlX2RmX2ZpbHRlcl8yICU+JQ0KICBkcGx5cjo6bGVmdF9qb2luKC4sIHNtcl9kZiwgYnkgPSAiaWQiKQ0KYGBgDQoNCiMjIFRyYW5zZm9ybWluZyBNT+KCgg0KDQpIZXJlIHdlIGFyZSB0cmFuc2Zvcm1pbmcgdGhlIE1P4oKCIHVuaXRzLiBUaGUgcmVzdWx0aW5nIHZhbHVlcyBhcmUgYXMgZm9sbG93czogIA0KDQotICoqTU8yKiogaXMgYWJzb2x1dGUgdmFsdWUgb2YgdGhlIGJhY2tncm91bmQgYW5kIGxlYWsgY29ycmVjdGVkIE1P4oKCIHNsb3BlIGZyb20gTGFiY2hhcnQgKG1vMmNvcnIpIHRpbWVzIHRoZSBuZXQgdm9sdW1lIG9mIHRoZSBjaGFtYmVyICh2b2x1bWUgLSBmaXNoIG1hc3MpLCDDlyA2MCwgw5cgNjAsIHRvIGFjaGlldmUgTU/igoIgYXMgbWdeLTFeIE/igoIgaF4tMV4NCg0KLSAqKk1PMl9nKiogaXMgTU8yIGRpdmlkZWQgYnkgZmlzaCBtYXNzIHRvIGFjaGlldmUgTU/igoIgYXMgbWdeLTFeIE/igoIgZ14tMV4gaF4tMV4gKGkuZS4gbWFzcyBzdGFuZGFyZGlzZWQpDQoNCi0gKipTTVIqKiBhYnNvbHV0ZSB2YWx1ZSBvZiB0aGUgU01SIGVzdGltYXRlcyB1c2luZyBtZXRob2RzIGRlc2NyaWJlZCBieSBDaGFib3QsIFN0ZWZmZW5zZW4gYW5kIEZhcnJlbGwgKDIwMTYpXlsxXV4gdGltZXMgdGhlIG5ldCB2b2x1bWUgb2YgdGhlIGNoYW1iZXIgKHZvbHVtZSAtIGZpc2ggbWFzcyksIMOXIDYwLCDDlyA2MCwgdG8gYWNoaWV2ZSBTTVIgYXMgbWdeLTFeIE/igoIgZ14tMV4gaF4tMV4pDQoNCi0gKipTTVJfZyoqIGlzIFNNUiBkaXZpZGVkIGJ5IGZpc2ggbWFzcyB0byBhY2hpZXZlIFNNUiBhcyBtZ14tMV4gT+KCgiBnXi0xXiBoXi0xXiAoaS5lLiBtYXNzIHN0YW5kYXJkaXNlZCkNCg0KLSAqKkRPKiogaXMgZGlzc29sdmVkIG94eWdlbiBwZXJjZW50YWdlIGNhbGN1bGF0ZWQgZnJvbSBP4oKCIHZhbHVlcyAobWdeLTFeIExeLTFeKSB1c2luZyB0aGUgcmVjb3JkZWQgdGVtcGVyYXR1cmUsIHNhbGluaXR5LCBhbmQgYSBjb25zdGFudCBhdG1vc3BoZXJpYyBwcmVzc3VyZSAoUGE7IDEwMTMuMjUpDQoNCi0gKipvMl9rcGEqKiBpcyB0aGUgT+KCgiBjb25jZW50cmF0aW9uIGluIGtpbG9wYXNjYWwgKGtwYSkuIFRoaXMgaXMgdXNlZCB0byBtYWtlIGEgY29tcGFyYXRpdmUgZmlndXJlIG9ubHkuICAgDQoNCg0KYGBge3J9DQojIENvbWJpbmUgYmFjayBpbnRvIG9uZSBkYXRhIGZyYW1lDQpzbG9wZV90aWR5IDwtIHNsb3BlX2RmX2ZpbHRlcl8zICU+JSANCiAgICBkcGx5cjo6bXV0YXRlKERPID0gY29udl9vMigNCiAgICAgICAgICAgICAgICAgICBvMiA9IG8yLA0KICAgICAgICAgICAgICAgICAgIGZyb20gPSAibWdfcGVyX2wiLA0KICAgICAgICAgICAgICAgICAgIHRvID0gInBlcmNlbnRfYS5zLiIsDQogICAgICAgICAgICAgICAgICAgdGVtcCA9IHRlbXAsICNDDQogICAgICAgICAgICAgICAgICAgc2FsID0gbWVhc3VyZWRfc2FsaW5pdHksDQogICAgICAgICAgICAgICAgICAgYXRtX3ByZXMgPSAxMDEzLjI1KSwNCiAgICAgICAgICAgICAgICAgIG8yX2twYSA9IGNvbnZfbzIoDQogICAgICAgICAgICAgICAgICAgbzIgPSBvMiwNCiAgICAgICAgICAgICAgICAgICBmcm9tID0gIm1nX3Blcl9sIiwNCiAgICAgICAgICAgICAgICAgICB0byA9ICJrUGEiLA0KICAgICAgICAgICAgICAgICAgIHRlbXAgPSB0ZW1wLCAjQw0KICAgICAgICAgICAgICAgICAgIHNhbCA9IG1lYXN1cmVkX3NhbGluaXR5LA0KICAgICAgICAgICAgICAgICAgIGF0bV9wcmVzID0gMTAxMy4yNSksDQogICAgICAgICAgICAgICAgICBuZXRfdm9sdW1lID0gdm9sdW1lIC0gbWFzcy8xMDAwLA0KICAgICAgICAgICAgICAgICAgTU8yID0gYWJzKG1vMmNvcnIpKm5ldF92b2x1bWUqNjAqNjAsDQogICAgICAgICAgICAgICAgICBNTzJfZyA9IE1PMi9tYXNzLA0KICAgICAgICAgICAgICAgICAgU01SID0gYWJzKHNtcl9jaGFib3QpKm5ldF92b2x1bWUqNjAqNjAsDQogICAgICAgICAgICAgICAgICBTTVJfZyA9IFNNUi9tYXNzDQogICAgICAgICAgICAgICAgICApDQpgYGANCg0KDQo8IS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0+DQojIPCfk4ogVmlzdWFsaXNpbmcgZGF0YQ0KPCEtLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tPg0KDQojIyBP4oKCIHZzIE1P4oKCDQoNCiMjIyBGaWd1cmUgUzENCg0KVGhpcyBpbnRlcmFjdGl2ZSB3YXMgdXNlZCB0byBpZGVudGlmeSBhbnkgb3V0bGllcnMsIG9yIHBvdGVudGlhbCBlcnJvcnMuICAgDQoNCmBgYHtyfQ0KbG1fbGluZXMgPC0gc2xvcGVfdGlkeSAlPiUNCiAgZ3JvdXBfYnkoaWQpICU+JQ0KICBzdW1tYXJpc2UoDQogICAgRE9fc2VxID0gbGlzdChzZXEobWluKERPKSwgbWF4KERPKSwgbGVuZ3RoLm91dCA9IDEwMCkpLCAgIyBHZW5lcmF0ZSBhIHNlcXVlbmNlIG9mIERPIHZhbHVlcw0KICAgIE1PMl9wcmVkID0gbGlzdChwcmVkaWN0KGxtKE1PMl9nIH4gRE8sIGRhdGEgPSBjdXJfZGF0YSgpKSwgbmV3ZGF0YSA9IGRhdGEuZnJhbWUoRE8gPSBzZXEobWluKERPKSwgbWF4KERPKSwgbGVuZ3RoLm91dCA9IDEwMCkpKSkNCiAgKSAlPiUNCiAgdW5uZXN0KGMoRE9fc2VxLCBNTzJfcHJlZCkpICAjIEV4cGFuZCBsaXN0cyBpbnRvIHJvd3MNCg0KIyBDcmVhdGUgc2NhdHRlciBwbG90IHdpdGggbWFya2VycyBmb3IgZWFjaCBmaXNoDQpwIDwtIHBsb3RfbHkoDQogIGRhdGEgPSBzbG9wZV90aWR5LA0KICB4ID0gfkRPLA0KICB5ID0gfk1PMl9nLA0KICB0eXBlID0gInNjYXR0ZXIiLA0KICBtb2RlID0gIm1hcmtlcnMiLA0KICBjb2xvciA9IH5pZCwgICMgQ29sb3VyIHBvaW50cyBieSBmaXNoIElEDQogIG1hcmtlciA9IGxpc3Qob3BhY2l0eSA9IDAuNiksDQogIG5hbWUgPSB+aWQNCikNCg0KIyBBZGQgcmVncmVzc2lvbiBsaW5lcyBmb3IgZWFjaCBmaXNoDQpwIDwtIHAgJT4lDQogIGFkZF90cmFjZSgNCiAgICBkYXRhID0gbG1fbGluZXMsDQogICAgeCA9IH5ET19zZXEsDQogICAgeSA9IH5NTzJfcHJlZCwNCiAgICB0eXBlID0gInNjYXR0ZXIiLA0KICAgIG1vZGUgPSAibGluZXMiLA0KICAgIGNvbG9yID0gfmlkLCAgIyBFbnN1cmUgZWFjaCBsaW5lIG1hdGNoZXMgaXRzIGNvcnJlc3BvbmRpbmcgZmlzaA0KICAgIGxpbmUgPSBsaXN0KHdpZHRoID0gMSwgZGFzaCA9ICJzb2xpZCIpLA0KICAgIHNob3dsZWdlbmQgPSBGQUxTRSAgIyBBdm9pZCBjbHV0dGVyaW5nIHRoZSBsZWdlbmQNCiAgKQ0KDQojIEZpbmFsIGxheW91dA0KcCA8LSBwICU+JQ0KICBsYXlvdXQoDQogICAgdGl0bGUgPSAiTU88c3ViPjI8L3N1Yj4gdnMgRGlzc29sdmVkIE94eWdlbiB3aXRoIGluZGl2aWR1YWwgbGluZWFyIHJlZ3Jlc3Npb25zIiwNCiAgICB4YXhpcyA9IGxpc3QodGl0bGUgPSAiRGlzc29sdmVkIE94eWdlbiAoJSkiKSwNCiAgICB5YXhpcyA9IGxpc3QodGl0bGUgPSAiTU88c3ViPjI8L3N1Yj4gKG1nPHN1cD4tMTwvc3VwPiBPPHN1Yj4yPC9zdWI+IGc8c3VwPi0xPC9zdXA+IGg8c3VwPi0xPC9zdXA+KSIpLA0KICAgIHNob3dsZWdlbmQgPSBGQUxTRQ0KICApDQoNCiMgRGlzcGxheSBwbG90DQpwDQpgYGANCg0KKioqRmlndXJlIFMxOioqKiBpbnRlcmFjdGl2ZSBwbG90IG9mIG1ldGFib2xpYyByYXRlIG1lYXN1cmVtZW50cyAoTU/igoI7IG1nIE/igoIgZ14tMV5oXi0xXikgYnkgZGlzc29sdmVkIG94eWdlbiBwZXJjZW50YWdlIChETykgZm9yIGFsbCBmaXNoLCBpbmNsdWRpbmcgYWxsIGVzdGltYXRlcyBkdXJpbmcgdGhlIFNNUiBwaGFzZSAoaS5lLiBpbnRlcm1pdHRlbnQgcGhhc2UpLiBJbmRpdmlkdWFsIGxpbmVhciByZWdyZXNzaW9uIHdlcmUgZml0dGVkIGZvciB2aXN1YWwgcmVmZXJlbmNlLCBhbmQgZG8gbm90IHJlcHJlc2VudCB0aGUgYmVzdCBmaXR0aW5nIHJlZ3Jlc3Npb24uIA0KDQoNCiMjIyBGaWd1cmUgUzINCg0KTG9va2luZyBhdCB0aGUgZGlmZmVyZW5jZSByZXNwb25zZXMgaW4gdGhlIHR3byBzYWxpbml0eSBncm91cHMuICAgDQoNCmBgYHtyfQ0Kc2FsaW5pdHlfc3VtbWFyeSA8LSBzbG9wZV90aWR5ICU+JSANCiAgZHBseXI6Omdyb3VwX2J5KHNhbGluaXR5X2dyb3VwKSAlPiUgDQogIGRwbHlyOjpyZWZyYW1lKG4gPSBsZW5ndGgodW5pcXVlKGlkKSkpDQoNCnNsb3BlX3RpZHkgJT4lIA0KICBnZ3Bsb3QoYWVzKHkgPSBNTzJfZywgeCA9IERPLCBjb2xvdXIgPSBpZCkpICsgIyBEZWZhdWx0IGFlc3RoZXRpY3MNCiAgZ2VvbV9wb2ludChzaG93LmxlZ2VuZCA9IEZBTFNFKSArDQogIGdlb21fc21vb3RoKGFlcyhncm91cCA9IGlkKSwgbWV0aG9kID0gImxtIiwgc2UgPSBGQUxTRSwgY29sb3VyID0gc2NhbGVzOjphbHBoYSgiYmxhY2siLCAwLjUpKSArICMgVHJhbnNwYXJlbnQgYmxhY2sgbGluZXMNCiAgZ2VvbV9zbW9vdGgobWV0aG9kID0gImxtIiwgc2UgPSBUUlVFLCBjb2xvdXIgPSAicmVkIikgKyAjIE92ZXJhbGwgc21vb3RoIGxpbmUNCiAgZ2VvbV9zbW9vdGgobWV0aG9kID0gImdhbSIsIGZvcm11bGEgPSB5IH4gcyh4LCBicyA9ICJjcyIpLCBzZSA9IFRSVUUsIGNvbG91ciA9ICJyZWQiLCBsaW5ldHlwZSA9ICJkYXNoZWQiKSArDQogIHRoZW1lX2NsZWFuKCkgKw0KICBmYWNldF93cmFwKH5zYWxpbml0eV9ncm91cCkgKw0KICBsYWJzKA0KICAgIHN1YnRpdGxlID0gIm1vMiB2cyBvMiBieSBzYWxpbml0eSB0cmVhdG1lbnQiLA0KICAgIHggPSAiRGlzc29sdmVkIG94eWdlbiBwZXJjZW50YWdlIChETykiLA0KICAgIHkgPSAiTU8yIChPMiBtZy9nL2gpIg0KICApICsNCiAgZ2VvbV90ZXh0KGRhdGEgPSBzYWxpbml0eV9zdW1tYXJ5LA0KICAgICAgICAgICAgYWVzKHggPSAtSW5mLCB5ID0gSW5mLCBsYWJlbCA9IHBhc3RlMCgiaXRhbGljKG4pID09ICIsIG4pKSwNCiAgICAgICAgICAgIGhqdXN0ID0gLTAuMSwgdmp1c3QgPSAxLjIsIGluaGVyaXQuYWVzID0gRkFMU0UsIHBhcnNlID0gVFJVRSkNCmBgYA0KDQoqKipGaWd1cmUgUzI6KioqIE1ldGFib2xpYyByYXRlIG1lYXN1cmVtZW50cyAoTU/igoI7IG1nIE/igoIgZ14tMV5oXi0xXikgYnkgZGlzc29sdmVkIG94eWdlbiBwZXJjZW50YWdlIChETykgZm9yIGZpc2ggZnJvbSB0aGUgdHdvIHNhbGluaXR5IHRyZWF0bWVudHMsIGluY2x1ZGluZyBhbGwgZXN0aW1hdGVzIGR1cmluZyB0aGUgU01SIHBoYXNlIChpLmUuIGludGVybWl0dGVudCBwaGFzZSkuIEluZGl2aWR1YWwgbGluZWFyIHJlZ3Jlc3Npb24gd2VyZSBmaXR0ZWQgZm9yIHZpc3VhbCByZWZlcmVuY2UsIGFuZCBkbyBub3QgcmVwcmVzZW50IHRoZSBiZXN0IGZpdHRpbmcgcmVncmVzc2lvbi4gVGhlIHNvbGlkIHJlZCBhbmQgZGFzaGVkIHJlZCBsaW5lIHJlcHJlc2VudHMgdGhlIGdsb2JhbCB0cmVuZCBpbiBlYWNoIG9mIHRoZSB0cmVhdG1lbnRzLCBib3RoIGEgbGluZWFyIChzb2xpZCByZWQpIGFuZCBub24tbGluZWFyIChkYXNoZWQgcmVkOyBHZW5lcmFsaXplZCBBZGRpdGl2ZSBNb2RlbCBmaXR0ZWQgd2l0aCBgZ2VvbV9zbW9vdGhgKS4NCg0KIyMjIEZpZ3VyZSBTMw0KDQpBIHBsb3QgdG8gbG9vayBhdCB0aGUgZGlmZmVyZW50IGNoYW1iZXIgdHlwZXMgICANCg0KYGBge3J9DQpjaGFtYmVyX3N1bW1hcnkgPC0gc2xvcGVfdGlkeSAlPiUgDQogIGRwbHlyOjpncm91cF9ieShjaGFtYmVyX3R5cGUpICU+JSANCiAgZHBseXI6OnJlZnJhbWUobiA9IGxlbmd0aCh1bmlxdWUoaWQpKSkNCg0Kc2xvcGVfdGlkeSAlPiUgDQogIGdncGxvdChhZXMoeSA9IE1PMl9nLCB4ID0gRE8sIGNvbG91ciA9IGlkKSkgKyAjIERlZmF1bHQgYWVzdGhldGljcw0KICBnZW9tX3BvaW50KHNob3cubGVnZW5kID0gRkFMU0UpICsNCiAgZ2VvbV9zbW9vdGgoYWVzKGdyb3VwID0gaWQpLCBtZXRob2QgPSAibG0iLCBzZSA9IEZBTFNFLCBjb2xvdXIgPSBzY2FsZXM6OmFscGhhKCJibGFjayIsIDAuNSkpICsgIyBUcmFuc3BhcmVudCBibGFjayBsaW5lcw0KICBnZW9tX3Ntb290aChtZXRob2QgPSAibG0iLCBzZSA9IFRSVUUsIGNvbG91ciA9ICJyZWQiKSArICMgT3ZlcmFsbCBzbW9vdGggbGluZQ0KICBnZW9tX3Ntb290aChzZSA9IFRSVUUsIGNvbG91ciA9ICJyZWQiLCBsaW5ldHlwZSA9ICJkYXNoZWQiKSArDQogIHRoZW1lX2NsZWFuKCkgKw0KICBmYWNldF93cmFwKH5jaGFtYmVyX3R5cGUsIHNjYWxlID0gImZyZWUiKSArDQogIGxhYnMoDQogICAgc3VidGl0bGUgPSAibW8yIHZzIG8yIGJ5IGNoYW1iZXIgdHlwZSIsDQogICAgeCA9ICJEaXNzb2x2ZWQgb3h5Z2VuIHBlcmNlbnRhZ2UgKERPKSIsDQogICAgeSA9ICJNTzIgKG1nIE8yIGcvaCkiDQogICkgKw0KICBnZW9tX3RleHQoZGF0YSA9IGNoYW1iZXJfc3VtbWFyeSwNCiAgICAgICAgICAgIGFlcyh4ID0gLUluZiwgeSA9IEluZiwgbGFiZWwgPSBwYXN0ZTAoIml0YWxpYyhuKSA9PSAiLCBuKSksDQogICAgICAgICAgICBoanVzdCA9IC0wLjEsIHZqdXN0ID0gMS4yLCBpbmhlcml0LmFlcyA9IEZBTFNFLCBwYXJzZSA9IFRSVUUpDQpgYGANCg0KKioqRmlndXJlIFMzOioqKiBNZXRhYm9saWMgcmF0ZSBtZWFzdXJlbWVudHMgKE1P4oKCOyBtZyBP4oKCIGdeLTFeaF4tMV4pIGJ5IGRpc3NvbHZlZCBveHlnZW4gcGVyY2VudGFnZSAoRE8pIGZvciBmaXNoIHRlc3RlZCBpbiB0aGUgNCBkaWZmZXJlbnQgY2hhbWJlciB0eXBlcywgaW5jbHVkaW5nIGFsbCBlc3RpbWF0ZXMgZHVyaW5nIHRoZSBTTVIgcGhhc2UgKGkuZS4gaW50ZXJtaXR0ZW50IHBoYXNlKS4gSW5kaXZpZHVhbCBsaW5lYXIgcmVncmVzc2lvbiB3ZXJlIGZpdHRlZCBmb3IgdmlzdWFsIHJlZmVyZW5jZSwgYW5kIGRvIG5vdCByZXByZXNlbnQgdGhlIGJlc3QgZml0dGluZyByZWdyZXNzaW9uLiBUaGUgc29saWQgcmVkIGFuZCBkYXNoZWQgcmVkIGxpbmUgcmVwcmVzZW50cyB0aGUgZ2xvYmFsIHRyZW5kIGluIGVhY2ggb2YgdGhlIHRyZWF0bWVudHMsIGJvdGggYSBsaW5lYXIgKHNvbGlkIHJlZCkgYW5kIG5vbi1saW5lYXIgKGRhc2hlZCByZWQ7IEdlbmVyYWxpemVkIEFkZGl0aXZlIE1vZGVsIGZpdHRlZCB3aXRoIGBnZW9tX3Ntb290aGApLg0KDQojIyMgRmlndXJlIFM0IA0KDQpIZXJlLCB3ZSBoYXZlIHJlY3JlYXRlZCBhIHNpbWlsYXIgcGxvdCB0byB0aGF0IHByZXNlbnRlZCBpbiBVcmJpbmEsIEdsb3ZlciwgYW5kIEZvcnN0ZXIgKDIwMTIpXlsxXV4gYW5kIGhhdmUgZXh0cmFjdGVkIHRoZSBtZWFuIGxldmVsIGRhdGEgZnJvbSBGaWd1cmUgMWEgdXNpbmcgdGhlIG1ldGFEaWdpdGlzZSBwYWNrYWdlIGluIFJeWzNdXi4gVGhpcyBkYXRhIGlzIGNhbGxlZCB1cmJpbmFfZXRfYWxfMjAxMi4gVGhpcyBhbGxvd3MgdXMgdG8gY29tcGFyZSB0aGUgZGlmZmVyZW5jZXMgaW4gTW8yIGFuZCB0aGUgcmVsYXRpb25zaGlwIGJldHdlZW4gTW8yIGFuZCBPPHN1Yj4yPC9zdWI+LiAgICANCg0KRmlyc3QgbWFraW5nIGEgYmlubmVkIGRhdGEgZnJhbWUgdG8gbWF0Y2ggVXJiaW5hLCBHbG92ZXIsIGFuZCBGb3JzdGVyICgyMDEyKSBhcyBjbG9zZWx5IGFzIHBvc3NpYmxlLiANCg0KYGBge3J9DQptaW5fbzJfa3BhIDwtIG1pbihzbG9wZV90aWR5JG8yX2twYSwgbmEucm0gPSBUUlVFKQ0KbWF4X28yX2twYSA8LSBtYXgoc2xvcGVfdGlkeSRvMl9rcGEsIG5hLnJtID0gVFJVRSkNCg0KdGltZV9iaW5fZGYgPC0gc2xvcGVfdGlkeSAlPiUNCiAgbXV0YXRlKG8yX2dyb3VwID0gY3V0KG8yX2twYSwgDQogICAgICAgICAgICAgICAgICAgICAgICBicmVha3MgPSBzZXEobWluX28yX2twYSwgbWF4X28yX2twYSwgbGVuZ3RoLm91dCA9IDEzKSwgIyAxMSBpbnRlcnZhbHMsIHNvIDEyIGJyZWFrcG9pbnRzDQogICAgICAgICAgICAgICAgICAgICAgICBsYWJlbHMgPSBwYXN0ZTAoIkdyb3VwICIsIDE6MTIpLCANCiAgICAgICAgICAgICAgICAgICAgICAgIGluY2x1ZGUubG93ZXN0ID0gVFJVRSkpICU+JSANCiAgZHBseXI6Omdyb3VwX2J5KG8yX2dyb3VwKSAlPiUgDQogIGRwbHlyOjpyZWZyYW1lKG1lYW5fTU8yX2cgPSBtZWFuKE1PMl9nKSozMS4yNSwNCiAgICAgICAgICAgICAgICAgbWVhbl9vMl9rcGEgPSBtZWFuKG8yX2twYSksDQogICAgICAgICAgICAgICAgIG4gPSBsZW5ndGgoTU8yX2cpKjMxLjI1LA0KICAgICAgICAgICAgICAgICBNTzJfZ19zZCA9IHNkKE1PMl9nKSozMS4yNSwNCiAgICAgICAgICAgICAgICAgbzJfa3BhX3NkID0gc2QobzJfa3BhKSkNCmBgYA0KICANCk5vdyB0aGUgcGxvdCB3aXRoIG91ciBtZWFuIGRhdGEgYW5kIHRoZSBtZWFuIGRhdGEgZnJvbSBVcmJpbmEsIEdsb3ZlciwgYW5kIEZvcnN0ZXIgKDIwMTIpXlsxXV4gICANCg0KYGBge3J9DQpuIDwtIHNsb3BlX3RpZHkgJT4lIA0KICBkcGx5cjo6ZGlzdGluY3QoaWQpICU+JSANCiAgbnJvdyguKQ0KDQojIEV4aXN0aW5nIHBsb3QNCnAgPC0gdGltZV9iaW5fZGYgJT4lIA0KICBnZ3Bsb3QoKSArDQogIGdlb21fcG9pbnQoZGF0YSA9IHNsb3BlX3RpZHksIA0KICAgICAgICAgICAgIGFlcyh5ID0gTU8yX2cgKiAzMS4yNSwgeCA9IG8yX2twYSksIA0KICAgICAgICAgICAgIHNpemUgPSAyLCBjb2xvciA9ICJncmV5IiwgYWxwaGEgPSAwLjMpICsgIA0KICBnZW9tX3BvaW50KGRhdGEgPSB0aW1lX2Jpbl9kZiwgDQogICAgICAgICAgICAgYWVzKHkgPSBtZWFuX01PMl9nLCB4ID0gbWVhbl9vMl9rcGEpLA0KICAgICAgICAgICAgIHNpemUgPSAzLCBjb2xvdXIgPSAiIzBFNEM5MiIsIHNob3cubGVnZW5kID0gRkFMU0UpICsNCiAgZ2VvbV9lcnJvcmJhcihkYXRhID0gdGltZV9iaW5fZGYsDQogICAgICAgICAgICAgICAgYWVzKHkgPSBtZWFuX01PMl9nLCB4ID0gbWVhbl9vMl9rcGEsDQogICAgICAgICAgICAgICAgICAgIHltaW4gPSBtZWFuX01PMl9nIC0gTU8yX2dfc2QsIHltYXggPSBtZWFuX01PMl9nICsgTU8yX2dfc2QpLCANCiAgICAgICAgICAgICAgICB3aWR0aCA9IDAuMTUsIGNvbG91ciA9ICIjMEU0QzkyIikgKw0KICBnZW9tX2Vycm9yYmFyaChkYXRhID0gdGltZV9iaW5fZGYsIA0KICAgICAgICAgICAgICAgICBhZXMoeSA9IG1lYW5fTU8yX2csIHggPSBtZWFuX28yX2twYSwNCiAgICAgICAgICAgICAgICAgICAgIHhtaW4gPSBtZWFuX28yX2twYSAtIG8yX2twYV9zZCwgeG1heCA9IG1lYW5fbzJfa3BhICsgbzJfa3BhX3NkKSwgDQogICAgICAgICAgICAgICAgIGhlaWdodCA9IDAuNCwgY29sb3VyID0gIiMwRTRDOTIiKSArDQogIGdlb21fcG9pbnQoZGF0YSA9IHVyYmluYV9ldF9hbF8yMDEyLCANCiAgICAgICAgICAgICBhZXMoeCA9IG8yX21lYW4sIHkgPSBtbzJfbWVhbiksIA0KICAgICAgICAgICAgIHNpemUgPSAzLCBzaGFwZSA9IDIxLCBmaWxsID0gIiNEMjFGM0MiLCBjb2xvciA9ICIjRDIxRjNDIiwgc3Ryb2tlID0gMSkgKw0KICBnZW9tX2Vycm9yYmFyKGRhdGEgPSB1cmJpbmFfZXRfYWxfMjAxMiwgDQogICAgICAgICAgICAgICAgYWVzKHggPSBvMl9tZWFuLCB5bWluID0gbW8yX21lYW4gLSBtbzJfc2QsIHltYXggPSBtbzJfbWVhbiArIG1vMl9zZCksIA0KICAgICAgICAgICAgICAgIHdpZHRoID0gMC4yLCBjb2xvdXIgPSAiI0QyMUYzQyIpICsNCiAgZ2VvbV9lcnJvcmJhcmgoZGF0YSA9IHVyYmluYV9ldF9hbF8yMDEyLCANCiAgICAgICAgICAgICAgICAgYWVzKHkgPSBtbzJfbWVhbiwgeG1pbiA9IG8yX21lYW4gLSBvMl9zZCwgeG1heCA9IG8yX21lYW4gKyBvMl9zZCksIA0KICAgICAgICAgICAgICAgICBoZWlnaHQgPSAwLjQsIGNvbG91ciA9ICIjRDIxRjNDIikgKw0KICBhbm5vdGF0ZSgidGV4dCIsIHggPSAwLCANCiAgICAgICAgICAgeSA9IDE2LCANCiAgICAgICAgICAgbGFiZWwgPSBicXVvdGUoYXRvcCgiQ3VycmVudCBkYXRhIChibHVlKSwgIiAqIGl0YWxpYyhuKSAqICIgPSAiICogLihuKSwNCiAgICAgICAgICAgICAgICAgICAgICAgICJVcmJpbmEgZGF0YSAocmVkKSwgIiAqIGl0YWxpYyhuKSAqICIgPSA2NyIpKSwgIA0KICAgICAgICAgaGp1c3QgPSAwLCB2anVzdCA9IDEsIHNpemUgPSA0KSArDQogIHRoZW1lX2NsZWFuKCkgKw0KICBsYWJzKA0KICAgIHN1YnRpdGxlID0gIiIsDQogICAgeCA9ICJQTzIgKGtQYSkiLA0KICAgIHkgPSAiTU8yICh1bW9sIE8yIGcvaCkiDQogICkgKw0KICBzY2FsZV95X2NvbnRpbnVvdXMobGltaXRzID0gYygwLCAxNiksIGJyZWFrcyA9IHNlcSgwLCAxNiwgYnkgPSAyKSkgKw0KICBzY2FsZV94X2NvbnRpbnVvdXMobGltaXRzID0gYygwLCAyMiksIGJyZWFrcyA9IHNlcSgwLCAyMiwgYnkgPSAyKSkgIA0KDQpwDQpgYGANCg0KKipGaWd1cmUgUzQ6KiogTWVhbiBhbmQgc3RhbmRhcmQgZXJyb3Igb2YgbWV0YWJvbGljIHJhdGUgKE1PPHN1Yj4yPC9zdWI+IM68bW9sIE88c3ViPjI8L3N1Yj4gZ14tMV4gaF4tMV4pIGFuZCBveHlnZW4gY29uY2VudHJhdGlvbiAoUE88c3ViPjI8L3N1Yj4ga1BhKSB1c2luZyAxMiBldmVubHkgc3BhY2VkIGJpbnMgb3ZlciB0aGUgTzxzdWI+Mjwvc3ViPiByYW5nZSBvZiBvYnNlcnZlZCBkYXRhIChibHVlIGRvdHMpLiBDb21wYXJlZCBhZ2FpbnN0IHRoZSBtZWFuIGFuZCBzdGFuZGFyZCBlcnJvciByZXBvcnRlZCBpbiBVcmJpbmEgKDIwMTIpXlsxXV4gKHJlZCBkb3RzKS4gVGhlIGdyZXkgZG90cyBhcmUgdGhlIHJhdyBvYnNlcnZlZCBkYXRhIGZvcm0gdGhlIHByZXNlbnQgc3R1ZHkuIA0KDQojIyBSb3V0aW5lIE1P4oKCDQoNCk1ha2luZyBhbiBTTVIgcGhhc2Ugb25seSBkYXRhIGZyYW1lICAgDQoNCmBgYHtyfQ0Kc2xvcGVfdGlkeV9zbXIgPC0gc2xvcGVfdGlkeSAlPiUgDQogIGRwbHlyOjpmaWx0ZXIocGhhc2UgPT0gInNtciIpDQpgYGANCg0KIyMjIEZpZ3VyZSBTNQ0KDQpSb3V0aW5lIE1PPHN1Yj4yPC9zdWI+IGJ5IHNhbGluaXR5ICAgDQoNCmBgYHtyfQ0KbWVhbl9tbzJfc2FsaW5pdHkgPC0gc2xvcGVfdGlkeV9zbXIgJT4lIA0KICBkcGx5cjo6Z3JvdXBfYnkoc2FsaW5pdHlfZ3JvdXApICU+JSANCiAgZHBseXI6OnJlZnJhbWUobWVhbl9tbzIgPSBtZWFuKE1PMiwgbmEucm0gPSBUUlVFKSkNCg0KZmlnX2kgPC0gZ2dwbG90KCkgKw0KICAgIGdlb21fdmlvbGluKGRhdGEgPSBzbG9wZV90aWR5X3NtciwgYWVzKHggPSBzYWxpbml0eV9ncm91cCwgeSA9IE1PMiwgZmlsbCA9IHNhbGluaXR5X2dyb3VwKSwgY29sb3IgPSBOQSwgYWxwaGEgPSAwLjMpICsNCiAgZ2VvbV9qaXR0ZXIoZGF0YSA9IHNsb3BlX3RpZHlfc21yLCBhZXMoeCA9IHNhbGluaXR5X2dyb3VwLCB5ID0gTU8yLCBmaWxsID0gc2FsaW5pdHlfZ3JvdXApLA0KICAgICAgICAgICAgICAgICAgICAgICBzaGFwZSA9IDIxLCBzaXplID0gMiwgY29sb3IgPSAiYmxhY2siLCBhbHBoYSA9IDAuMikgKw0KICBnZW9tX2JveHBsb3QoZGF0YSA9IHNsb3BlX3RpZHlfc21yLCBhZXMoeCA9IHNhbGluaXR5X2dyb3VwLCB5ID0gTU8yLCBmaWxsID0gc2FsaW5pdHlfZ3JvdXApLA0KICAgICAgICAgICAgICAgICAgICAgICAgc2l6ZSA9IDEsIGFscGhhID0gMC41LCBvdXRsaWVyLnNoYXBlID0gTkEsIHdpZHRoID0gMC4zKSArDQogIGdlb21fcG9pbnQoZGF0YSA9IG1lYW5fbW8yX3NhbGluaXR5LCANCiAgICAgICAgICAgICAgICBhZXMoeCA9IHNhbGluaXR5X2dyb3VwLCB5ID0gbWVhbl9tbzIsIGZpbGwgPSBzYWxpbml0eV9ncm91cCksIA0KICAgICAgICAgICAgICAgIHNpemUgPSAzLCBhbHBoYSA9IDAuOCwgY29sb3VyID0gImJsYWNrIiwgc3Ryb2tlID0gMikgKw0KICBzY2FsZV9maWxsX21hbnVhbCh2YWx1ZXMgPSBjKCIjNEI1MzIwIiwgIiMwMDAwODAiKSkgKyAgIyBDdXN0b20gZmlsbCBjb2xvdXJzDQogIHNjYWxlX2NvbG91cl9tYW51YWwodmFsdWVzID0gYygiIzRCNTMyMCIsICIjMDAwMDgwIikpICsNCiAgdGhlbWVfY2xlYW4oKSArDQogIHRoZW1lKGxlZ2VuZC5wb3NpdGlvbiA9ICJub25lIikgKw0KICBsYWJzKA0KICAgIHN1YnRpdGxlID0gIiIsDQogICAgeCA9ICJTYWxpbml0eSBncm91cCAocHB0KSIsDQogICAgeSA9ICJSb3V0aW5lIE1PMiAobWcgTzIgZy9oKSINCiAgKQ0KDQpmaWdfaQ0KYGBgDQoNCioqRmlndXJlIFM1OioqIFBsb3Qgb2YgYWxsIE1PPHN1Yj4yPC9zdWI+IG1lYXN1cmVzIGR1cmluZyBTTVIgcGhhc2UgYnkgc2FsaW5pdHkgdHJlYXRtZW50LiBUaGUgc21hbGwgcG9pbnRzIGFyZSB0aGUgcmF3IG9ic2VydmVkIHZhbHVlcywgdGhlIHNoYWRlZCBhcmVhIGJlaGluZCB0aGUgcG9pbnRzIGlzIHRoZSBhIGtlcm5lbCBkZW5zaXR5IG9mIHRoZSBvYnNlcnZlZCBkYXRhLCB0aGUgYm94IHBsb3Qgc2hvd3MgdGhlIG1lZGlhbiBhbmQgaW50ZXJxdWFydGlsZSByYW5nZSAoSVFSKSwgYW5kIHRoZSBsYXJnZSBwb2ludCBzaG93cyB0aGUgbWVhbi4gDQoNCiMjIFNNUg0KDQojIyMgRmlndXJlIFM2DQoNCkhlcmUncyB0aGUgc2FtZSBwbG90IGJ1dCBmb3Igb25seSB0aGUgU01SLCBhcyBlc3RpbWF0ZWQgd2l0aCBjYWxjU01SIGZ1bmN0aW9uIGJ5IENoYWJvdCwgU3RlZmZlbnNlbiBhbmQgRmFycmVsbCAoMjAxNileWzFdXi4gU3BlY2lmaWNhbGx5LCB3ZSB1c2UgbWVhbiBvZiB0aGUgbG93ZXN0IG5vcm1hbCBkaXN0cmlidXRpb24gKE1MTkQpIHdoZXJlIENWbWxuZCA8IDUuNCBhbmQgdGhlIG1lYW4gb2YgdGhlIGxvd2VyIDIwJSBxdWFudGlsZSAocTAuMikgd2VyZSBDVm1sbmQgPiA1LjQuIElmIENWbWxuZCBpcyBub3QgY2FsY3VsYXRlZCB3ZSBoYXZlIHVzZWQgcTAuMi4gICANCg0KYGBge3J9DQpzbXJfb25seSA8LSBzbG9wZV90aWR5X3NtciAlPiUgDQogIGRwbHlyOjpncm91cF9ieShpZCkgJT4lIA0KICBkcGx5cjo6c2xpY2UoMSkNCg0KbWVhbl9zbXJfc2FsaW5pdHkgPC0gc21yX29ubHkgJT4lIA0KICBkcGx5cjo6Z3JvdXBfYnkoc2FsaW5pdHlfZ3JvdXApICU+JSANCiAgZHBseXI6OnJlZnJhbWUobWVhbl9zbXIgPSBtZWFuKFNNUiwgbmEucm0gPSBUUlVFKSkNCg0KZmlnX2kgPC0gZ2dwbG90KCkgKw0KICAgIGdlb21fdmlvbGluKGRhdGEgPSBzbXJfb25seSwgYWVzKHggPSBzYWxpbml0eV9ncm91cCwgeSA9IFNNUiwgZmlsbCA9IHNhbGluaXR5X2dyb3VwKSwgY29sb3IgPSBOQSwgYWxwaGEgPSAwLjMpICsNCiAgZ2VvbV9qaXR0ZXIoZGF0YSA9IHNtcl9vbmx5LCBhZXMoeCA9IHNhbGluaXR5X2dyb3VwLCB5ID0gU01SLCBmaWxsID0gc2FsaW5pdHlfZ3JvdXApLA0KICAgICAgICAgICAgICAgICAgICAgICBzaGFwZSA9IDIxLCBzaXplID0gMiwgY29sb3IgPSAiYmxhY2siLCBhbHBoYSA9IDAuMikgKw0KICBnZW9tX2JveHBsb3QoZGF0YSA9IHNtcl9vbmx5LCBhZXMoeCA9IHNhbGluaXR5X2dyb3VwLCB5ID0gU01SLCBmaWxsID0gc2FsaW5pdHlfZ3JvdXApLA0KICAgICAgICAgICAgICAgICAgICAgICAgc2l6ZSA9IDEsIGFscGhhID0gMC41LCBvdXRsaWVyLnNoYXBlID0gTkEsIHdpZHRoID0gMC4zKSArDQogIGdlb21fcG9pbnQoZGF0YSA9IG1lYW5fc21yX3NhbGluaXR5LCANCiAgICAgICAgICAgICAgICBhZXMoeCA9IHNhbGluaXR5X2dyb3VwLCB5ID0gbWVhbl9zbXIsIGZpbGwgPSBzYWxpbml0eV9ncm91cCksIA0KICAgICAgICAgICAgICAgIHNpemUgPSAzLCBhbHBoYSA9IDAuOCwgY29sb3VyID0gImJsYWNrIiwgc3Ryb2tlID0gMikgKw0KICBzY2FsZV9maWxsX21hbnVhbCh2YWx1ZXMgPSBjKCIjNEI1MzIwIiwgIiMwMDAwODAiKSkgKyAgIyBDdXN0b20gZmlsbCBjb2xvdXJzDQogIHNjYWxlX2NvbG91cl9tYW51YWwodmFsdWVzID0gYygiIzRCNTMyMCIsICIjMDAwMDgwIikpICsNCiAgdGhlbWVfY2xlYW4oKSArDQogIHRoZW1lKGxlZ2VuZC5wb3NpdGlvbiA9ICJub25lIikgKw0KICBsYWJzKA0KICAgIHN1YnRpdGxlID0gIiIsDQogICAgeCA9ICJTYWxpbml0eSBncm91cCAocHB0KSIsDQogICAgeSA9ICJNTzIgKG1nIE8yIGcvaCkiDQogICkNCg0KZmlnX2kNCmBgYA0KDQoqKkZpZ3VyZSBTNjoqKiBUaGUgc3RhbmRhcmQgbWV0YWJvbGljIHJhdGUgKFNNUikgYnkgc2FsaW5pdHkgdHJlYXRtZW50LiBUaGUgc21hbGwgcG9pbnRzIGFyZSB0aGUgcmF3IG9ic2VydmVkIHZhbHVlcywgdGhlIHNoYWRlZCBhcmVhIGJlaGluZCB0aGUgcG9pbnRzIGlzIHRoZSBhIGtlcm5lbCBkZW5zaXR5IG9mIHRoZSBvYnNlcnZlZCBkYXRhLCB0aGUgYm94IHBsb3Qgc2hvd3MgdGhlIG1lZGlhbiBhbmQgaW50ZXJxdWFydGlsZSByYW5nZSAoSVFSKSwgYW5kIHRoZSBsYXJnZSBwb2ludCBzaG93cyB0aGUgbWVhbi4gDQoNCiMjIEluZGl2aWR1YWwgT+KCgiwgTU/igoIsIGFuZCBTTVINCg0KSGVyZSB3ZSB3aWxsIHBsb3QgdGhlIGluZGl2aWR1YWwgcmVsYXRpb25zaGlwIGJldHdlZW4gT+KCgiwgTU/igoIsIGFuZCBTTVIgZm9yIGFsbCBmaXNoICAgDQoNCkNyZWF0ZSBvdXRwdXQgZGlyZWN0b3J5IGlmIG5lZWRlZCAgDQoNCmBgYHtyLCBlY2hvPUZBTFNFfQ0KIyBDcmVhdGUgb3V0cHV0IGRpcmVjdG9yeSBpZiBuZWVkZWQNCm91dHB1dF9maWdfc2xvcGVzX3dkIDwtIGZpbGUucGF0aChvdXRwdXRfZmlnX3dkLCAic2xvcGVzIikNCmlmICghZGlyLmV4aXN0cyhvdXRwdXRfZmlnX3Nsb3Blc193ZCkpIHsNCiAgZGlyLmNyZWF0ZShvdXRwdXRfZmlnX3Nsb3Blc193ZCkNCn0NCmBgYA0KDQojIyMgRmlndXJlIFM3DQoNCkxvb3AgdGhyb3VnaCBlYWNoIGZpc2ggSUQgdG8gY3JlYXRlIGEgcGxvdCwgc2F2ZSB0aGVzZSB0byBhIHNpbmdsZSBQREYgZmlsZQ0KDQpgYGB7ciwgZmlnLmhlaWdodCA9IDMsIGZpZy53aWR0aD0gNH0NCmxpYnJhcnkoZ2dwbG90MikNCmxpYnJhcnkoZHBseXIpDQoNCiMgRGVmaW5lIGNvbnNpc3RlbnQgY29sb3JzIGZvciBhbGwgcGhhc2VzDQpwaGFzZV9jb2xvcnMgPC0gYygNCiAgIjUwYyIgID0gIiNlNTZiNmYiLA0KICAiNzVjIiAgPSAiI2I1NjU3NiIsDQogICIxMDBjIiA9ICIjNmQ1OTdhIiwNCiAgInNtciIgID0gIiMzNTUwNzAiDQopDQoNCmlkcyA8LSBzbG9wZV90aWR5ICU+JSANCiAgZHBseXI6OmRpc3RpbmN0KGlkKSAlPiUgDQogIHB1bGwoaWQpICU+JSANCiAgYXMubGlzdCgpDQoNCk1PMl9wbG90X2xpc3QgPC0gbGlzdCgpDQoNCiMgMSkgT3BlbiB0aGUgUERGIGRldmljZSBvbmNlDQpwZGYoDQogIGZpbGUgICA9IGZpbGUucGF0aChvdXRwdXRfZmlnX3Nsb3Blc193ZCwgImNvbWJpbmVkX3Nsb3Blcy5wZGYiKSwgDQogIHdpZHRoICA9IDgsIA0KICBoZWlnaHQgPSA2DQopDQoNCiMgMikgTG9vcCBvdmVyIElEcyBhbmQgY3JlYXRlIGVhY2ggcGxvdA0KZm9yIChpZF9pIGluIGlkcykgew0KICANCiAgc21yIDwtIHNsb3BlX3RpZHkgJT4lIA0KICAgIGRwbHlyOjpmaWx0ZXIoaWQgPT0gaWRfaSkgJT4lIA0KICAgIGRwbHlyOjpzbGljZSgxKSAlPiUgDQogICAgZHBseXI6OnB1bGwoU01SKQ0KICANCiAgcGxvdCA8LSBzbG9wZV90aWR5ICU+JSANCiAgICBkcGx5cjo6ZmlsdGVyKGlkID09IGlkX2kpICU+JSANCiAgICBnZ3Bsb3QoYWVzKHggPSBvMiwgeSA9IE1PMikpICsNCiAgICBnZW9tX2hsaW5lKHlpbnRlcmNlcHQgPSBzbXIsIGxpbmV0eXBlID0gImRhc2hlZCIsIGNvbG9yID0gImRhcmtyZWQiKSArDQogICAgZ2VvbV9wb2ludChhZXMoY29sb3VyID0gcGhhc2UpKSArDQogICAgc2NhbGVfY29sb3JfbWFudWFsKHZhbHVlcyA9IHBoYXNlX2NvbG9ycywgZHJvcCA9IEZBTFNFKSArICAjIEVuc3VyZXMgY29uc2lzdGVudCBjb2xvdXJzDQogICAgdGhlbWVfY2xlYW4oKSArDQogICAgbGFicygNCiAgICAgIHN1YnRpdGxlID0gcGFzdGUwKGlkX2ksICIgc2xvcGVzIiksDQogICAgICB4ID0gIk1lYW4gbzIgKG1nX3Blcl9sKSIsDQogICAgICB5ID0gImFicyhtbzIpIChtZ19wZXJfbCkiDQogICAgKQ0KICANCiAgIyBJbnN0ZWFkIG9mIHNhdmluZyBlYWNoIHBsb3Qgc2VwYXJhdGVseSwganVzdCBwcmludCBpdA0KICBwcmludChwbG90KQ0KICANCiAgTU8yX3Bsb3RfbGlzdFtbaWRfaV1dIDwtIHBsb3QNCn0NCg0KIyAzKSBDbG9zZSB0aGUgUERGIGRldmljZSAqYWZ0ZXIqIHRoZSBsb29wDQpkZXYub2ZmKCkNCg0KZm9yIChwIGluIE1PMl9wbG90X2xpc3QpIHsNCiAgcHJpbnQocCkNCn0NCmBgYA0KDQoqKkZpZ3VyZSBTNyoqOiBNZXRhYm9saWMgcmF0ZSBtZWFzdXJlbWVudHMgKE1P4oKCOyBtZyBP4oKCIGdeLTFeaF4tMV4gYnkgTzxzdWI+Mjwvc3ViPikgYXQgZWFjaCBvZiB0aGUgZm91ciBleHBlcmltZW50YWwgcGhhc2UgKHRoZSBvdmVyIG5pZ2h0IFNNUiBpbnRlcm1pdHRlbnQtZmxvdyByZXNwaXJvbWV0cnkgcGhhc2UsIGNsb3NlZCBwaGFzZXMgYXQgNTAlLCA3NSUgb3IgMTAwJSAgTzxzdWI+Mjwvc3ViPikuIFRoZSBlc3RpbWF0ZWQgU01SIHZhbHVlIGlzIHJlcHJlc2VudGVkIGFzIGEgZGFzaGVkIHJlZCBsaW5lLg0KDQo8IS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0+DQojIPCfp64gQW5hbHlzaXMgDQo8IS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0+DQoNCiMjIFJvdXRpbmUgTU8yDQoNCiMjIyBTY2FsaW5nIHByZWRpY3RvcnMNCg0KSGVyZSB3ZSBzY2FsZSBvdXIgcHJlZGljdG9ycyBmb3IgdGhlIG1vZGVsICAgDQoNCmBgYHtyfQ0Kc2NhbGVfbGlzdCA8LSBjKCJ0ZW1wIiwgIm9yZGVyIiwgIm1hc3MiKSANCg0Kc2xvcGVfdGlkeV9zbXIgPC0gc2xvcGVfdGlkeV9zbXIgJT4lDQogIGRwbHlyOjptdXRhdGUoYWNyb3NzKGFsbF9vZihzY2FsZV9saXN0KSwgfiBzY2FsZSgueCwgY2VudGVyID0gVFJVRSwgc2NhbGUgPSBGQUxTRSksDQogICAgICAgICAgICAgICAgICAgICAgIC5uYW1lcyA9ICJ7LmNvbH1feiIpLA0KICAgICAgICAgICAgICAgIGxpZ2h0X2RhcmtfYyA9IGlmX2Vsc2UobGlnaHRfZGFyayA9PSAibGlnaHQiLCAwLjUsIC0wLjUpKQ0KYGBgDQoNCiMjIyBNb2RlbCBzdHJ1Y3R1cmUNCg0KSGVyZSB3ZSB3aWxsIHVzZSBhIEJheWVzaWFuIEdlbmVyYWxpc2VkIExpbmVhciBNaXhlZCBNb2RlbCAoR0xNTSkgd2l0aCBhIEdhbW1hIGRpc3RyaWJ1dGlvbiBhbmQgYSBsb2cgbGluaywgd2hlcmUgdGhlIHNoYXBlIHBhcmFtZXRlciAoSykgaXMgYWxzbyBtb2RlbGxlZCBhcyBhIGZ1bmN0aW9uIG9mIHByZWRpY3RvcnMuIFRoaXMgbW9kZWxzIE1PPHN1Yj4yPC9zdWI+IGJ5IHNhbGluaXR5IGR1cmluZyB0aGUgU01SIHBoYXNlIHRvIHNlZSBpZiB0aGUgZmlzaCBoZWxkIGF0IGRpZmZlcmVudCBzYWxpbml0aWVzIGhhdmUgZGlmZmVyZW50IFNNUnMuIFdlIGhhdmUgYWxzbyBhZGRlZCBhIGZldyBzY2FsZWQgcHJlZGljdG9ycywgdGhhdCBtYXkgaGVscCBkZXNjcmliZSB2YXJpYXRpb24gaW4gdGhlIGRhdGEsIHN1Y2ggYXMgbWFzcyAoZzsgMC4yMeKAkzEuNikgdGVtcGVyYXR1cmUgKMKwQzsgMTMuODQx4oCTMTQuMjc3KSwgbWVhc3VyZW1lbnQgb3JkZXIgKDHigJMyOCksIGFuZCBsaWdodC9kYXJrIGN5Y2xlIChsaWdodCBvciBkYXJrOyBsaWdodCBiZXR3ZWVuIDA3OjAwOjAwIGFuZCAxOTowMDowMCksIHdlIGFsc28gaW5jbHVkZSBhIHJhbmRvbSBlZmZlY3QgZm9yIGZpc2ggaWQgdG8gYWNjb3VudCBmb3IgbXVsdGlwbGUgTU88c3ViPjI8L3N1Yj4gbWVhc3VyZXMgb24gZWFjaCBmaXNoICgxIC0gNTgpLiBXZSBhbGxvd2VkIHRoZSB0aGUgc2hhcGUgcGFyYW1ldGVyIChLKSB0byB2YXJ5IGFzIGEgZnVuY3Rpb24gb2Ygc29tZSBvZiB0aGUgcHJlZGljdG9ycyAoZS5nLiBzYWxpbml0eV9ncm91cCwgb3JkZXJfeikgdG8gaW1wcm92ZSBmaXQuICAgDQoNCmBgYHtyfQ0KbW8yX2dhbW1hX2JmIDwtIGJmKE1PMiB+IHRlbXBfeiArIA0KICAgICAgICAgICAgICAgICAgICAgb3JkZXJfeiAgKyANCiAgICAgICAgICAgICAgICAgICAgIGxpZ2h0X2RhcmtfYyArIA0KICAgICAgICAgICAgICAgICAgICAgbWFzc196ICsgDQogICAgICAgICAgICAgICAgICAgICBzYWxpbml0eV9ncm91cCArICgxfGlkKSwNCiAgICAgICAgICAgICAgICAgICBzaGFwZSB+IHNhbGluaXR5X2dyb3VwICsgDQogICAgICAgICAgICAgICAgICAgICBvcmRlcl96LA0KICAgICAgICAgICAgICAgICAgIGZhbWlseSA9IEdhbW1hKGxpbmsgPSAibG9nIikpDQpgYGANCg0KDQojIyMgUHJpb3Igc2VsZWN0aW9uDQoNClRoZXNlIGFyZSB0aGUgZGVmYXVsdCBwcmlvcnMuIFdlIHdpbGwgdXNlIHRoZXNlLg0KDQpgYGB7cn0NCmRlZmF1bHRfcHJpb3IgPC0gZ2V0X3ByaW9yKG1vMl9nYW1tYV9iZiwgZGF0YSA9IHNsb3BlX3RpZHlfc21yLCBmYW1pbHkgPSBHYW1tYShsaW5rID0gImxvZyIpKQ0KZGVmYXVsdF9wcmlvciAlPiUgDQogIGd0KCkNCmBgYA0KDQojIyMgUnVuIG1vZGVsDQoNCkhlcmUgd2UgcnVuIHRoZSBtb2RlbCwgSSBoYXZlIGhhc2hlZCB0aGlzIG91dCBiZWNhdXNlIEkgaGF2ZSBzYXZlZCB0aGUgbW9kZWwgZm9yIHF1aWNrIHJlbG9hZGluZy4gDQoNCuKPre+4jyAqKlNraXAgdGhpcyBzdGVwKiogaWYgeW91IGhhdmUgZG93bmxvYWRlZCB0aGUgc2F2ZWQgbW9kZWwgJ21vMl9tb2RfZ2FtbWEucmRzJw0KDQpgYGB7ciwgZWNobz1GQUxTRX0NCiMgc2V0d2Qob3V0cHV0X21vZHNfd2QpDQojIA0KIyBtbzJfbW9kX2dhbW1hIDwtIGJybShtbzJfZ2FtbWFfYmYsDQojICAgICAgICAgICAgICAgIGRhdGEgPSBzbG9wZV90aWR5X3NtciwNCiMgICAgICAgICAgICAgICAgY29yZXMgPSA0LA0KIyAgICAgICAgICAgICAgICBjaGFpbnMgPSA0LA0KIyAgICAgICAgICAgICAgICBwcmlvciA9IGRlZmF1bHRfcHJpb3IsDQojICAgICAgICAgICAgICAgIHdhcm11cCA9IDEwMDAsDQojICAgICAgICAgICAgICAgIHNlZWQgPSAxNDMwMTksDQojICAgICAgICAgICAgICAgIHRoaW4gPSAyLA0KIyAgICAgICAgICAgICAgICBpdGVyID0gODAwMCwNCiMgICAgICAgICAgICAgICAgc2F2ZV9wYXJzID0gc2F2ZV9wYXJzKGFsbD1UUlVFKSwNCiMgICAgICAgICAgICAgICAgc2FtcGxlX3ByaW9yID0gVFJVRSwNCiMgICAgICAgICAgICAgICAgZmlsZSA9ICdtbzJfbW9kX2dhbW1hJykNCiMgcHJpbnQoIk1vZGVsIGNvbXBsZXRlIikNCmBgYA0KDQpIZXJlIHdlIHJlbG9hZCB0aGUgbW9kZWwgICANCg0KYGBge3J9DQpzZXR3ZChvdXRwdXRfbW9kc193ZCkNCg0KbW8yX21vZF9nYW1tYSA8LSAgcmVhZFJEUyhmaWxlID0gIm1vMl9tb2RfZ2FtbWEucmRzIikNCmBgYA0KDQojIyMgTW9kZWwgZGlhZ25vc3RpY3MNCg0KQ2hlY2tpbmcgbW9kZWwgY29udmVyZ2VuY2UgICANCg0KYGBge3J9DQpjb2xvcl9zY2hlbWVfc2V0KCJyZWQiKQ0KcGxvdChtbzJfbW9kX2dhbW1hLCBhc2sgPSBGKQ0KYGBgDQoNCkNoZWNraW5nIHJoYXQgYXJlIGVxdWFsIHRvIG9uZSAgIA0KDQpgYGB7cn0NCnN1bW1hcnkobW8yX21vZF9nYW1tYSkNCmBgYA0KDQoNClVzaW5nIGxlYXZlIG9uZSBvdXQgKGxvbykgbWVhc3VyZSBvZiBmaXQsIHRoZSBtb2RlbCBhcHBlYXJzIHRvIHByZWZvcm0gd2VsbCwgYWxsIFBhcmV0byBrIGVzdGltYXRlcyBhcmUgZ29vZCAoayA8IDAuNykgICANCg0KYGBge3J9DQpsb28obW8yX21vZF9nYW1tYSkNCmBgYA0KDQpNb2RlbCBwcmVkaWN0aW9ucyBnZW5lcmFsbHkgYWxpZ24gd2l0aCB0aGUgb2JzZXJ2ZWQgZGF0YSAgIA0KDQpgYGB7cn0NCmNvbG9yX3NjaGVtZV9zZXQoInJlZCIpDQpwIDwtIHBwX2NoZWNrKG1vMl9tb2RfZ2FtbWEsIHR5cGUgPSAiZGVuc19vdmVybGF5IikNCnANCmBgYA0KDQoNCiMjIyDwn5OIIFJlc3VsdHMNCg0KV2UgZGlkIG5vdCBzZWUgYSBtZWFuaW5nZnVsIGRpZmZlcmVuY2UgYmV0d2VlbiB0aGUgcm91dGluZSBtZXRhYm9saWMgcmF0ZSBmb3IgZmlzaCBmcm9tIHRoZSB0d28gc2FsaW5pdHkgdHJlYXRtZW50cy4gICANCg0KIyMjIyBUYWJsZSBTMQ0KDQoqKlRhYmxlIFMxKio6IEZpeGVkIGVmZmVjdCBFc3RpbWF0ZXMgKM6yKSBhbmQgOTUlIENyZWRpYmxlIEludGVydmFscyAoOTUlIENJKSAgIA0KDQpgYGB7cn0NCm1vZGVsX2VzdCA8LSBmaXhlZihtbzJfbW9kX2dhbW1hLCBwcm9icyA9IGMoMC4wMjUsIDAuOTc1KSkgJT4lIA0KICBhcy5kYXRhLmZyYW1lKCkgJT4lIA0KICB0aWJibGU6OnJvd25hbWVzX3RvX2NvbHVtbih2YXIgPSAiUHJlZGljdG9yIikgJT4lIA0KICBkcGx5cjo6bXV0YXRlKCfOsicgPSByb3VuZChFc3RpbWF0ZSwgMyksDQogICAgICAgICAgICAgICAgUTIuNSA9IHJvdW5kKFEyLjUsIDMpLA0KICAgICAgICAgICAgICAgIFE5Ny41ID0gcm91bmQoUTk3LjUsIDMpLA0KICAgICAgICAgICAgICAgICc5NSUgQ0knID0gcGFzdGUwKCJbIiwgUTIuNSwgIiwgIiwgUTk3LjUsICJdIikpDQoNCm1vZGVsX2VzdCAlPiUgDQogIGRwbHlyOjpzZWxlY3QoUHJlZGljdG9yLCAnzrInLCAnOTUlIENJJykgJT4lIA0KICBndCgpDQpgYGANCiAgICAgDQpMb29raW5nIGF0IHRoZSBtYXJnaW5hbCBtZWFuIGRpZmZlcmVuY2UgYmV0d2VlbiBzYWxpbml0eSBncm91cHMgICANCg0KYGBge3J9DQplbV9yZXN1bHRzIDwtIGVtbWVhbnMobW8yX21vZF9nYW1tYSwgfiBzYWxpbml0eV9ncm91cCkgDQpjb250cmFzdF9yZXN1bHRzIDwtIGNvbnRyYXN0KGVtX3Jlc3VsdHMsIG1ldGhvZCA9ICJwYWlyd2lzZSIpDQplbV9yZXN1bHRzX2RmIDwtICBlbV9yZXN1bHRzICU+JSB0aWR5KCkgJT4lIA0KICBtdXRhdGUoYWNyb3NzKHdoZXJlKGlzLm51bWVyaWMpLCB+IGV4cCguKSkpDQpjb250cmFzdF9yZXN1bHRzX2RmIDwtICBjb250cmFzdF9yZXN1bHRzICU+JSB0aWR5KCkgJT4lIA0KICBtdXRhdGUoYWNyb3NzKHdoZXJlKGlzLm51bWVyaWMpLCB+IGV4cCguKSkpDQoNCg0KZW1fcmVzdWx0c19kZiAlPiUgDQogIGd0KCkNCmBgYA0KDQpQdWxsaW5nIHRoZSBlbW1lYW5zIGRyYXdzIGZvciBvdXIgcGxvdCAgIA0KDQpgYGB7cn0NCmVtbWVhbnNfZHJhd3MgPC0gbW8yX21vZF9nYW1tYSAlPiUNCiAgZW1tZWFucyh+IHNhbGluaXR5X2dyb3VwKSAlPiUNCiAgZ2F0aGVyX2VtbWVhbnNfZHJhd3MoKSAlPiUgDQogIGRwbHlyOjptdXRhdGUoLnZhbHVlID0gIGV4cCgudmFsdWUpLA0KICAgICAgICAgICAgICAgIHNhbGluaXR5X2dyb3VwID0gYXMuY2hhcmFjdGVyKHNhbGluaXR5X2dyb3VwKSkNCg0KZW1tZWFuc19jb250cmFzdF9kcmF3cyA8LSAgbW8yX21vZF9nYW1tYSAlPiUNCiAgZW1tZWFucyh+IHNhbGluaXR5X2dyb3VwKSAlPiUNCiAgY29udHJhc3QobWV0aG9kID0gInBhaXJ3aXNlIikgJT4lDQogIGdhdGhlcl9lbW1lYW5zX2RyYXdzKCkgJT4lIA0KICBkcGx5cjo6bXV0YXRlKC52YWx1ZSA9ICBleHAoLnZhbHVlKSkNCmBgYA0KDQojIyMjIEZpZ3VyZSBTOA0KDQoqKipOT1RFOiBUaGlzIHBsb3QgaXMgaW4gdGhlIG1haW4gdGV4dCBvZiB0aGUgbWFudXNjcmlwdCBhcyBGaWd1cmUgMWEqKioNCg0KYGBge3J9DQptZWFuX21vMl9zYWxpbml0eSA8LSBzbG9wZV90aWR5X3NtciAlPiUgDQogIGRwbHlyOjpncm91cF9ieShzYWxpbml0eV9ncm91cCkgJT4lIA0KICBkcGx5cjo6cmVmcmFtZShtZWFuX21vMiA9IG1lYW4oTU8yLCBuYS5ybSA9IFRSVUUpKQ0KDQoNCmZpZ18xIDwtIGdncGxvdCgpICsNCiAgZ2VvbV92aW9saW4oZGF0YSA9IHNsb3BlX3RpZHlfc21yLA0KICAgICAgICAgICAgICBhZXMoeCA9IHNhbGluaXR5X2dyb3VwLCB5ID0gTU8yLCBmaWxsID0gc2FsaW5pdHlfZ3JvdXApLA0KICAgICAgICAgICAgICBjb2xvciA9IE5BLCBhbHBoYSA9IDAuMikgKw0KICBnZW9tX2ppdHRlcihkYXRhID0gc2xvcGVfdGlkeV9zbXIsDQogICAgICAgICAgICAgIGFlcyh4ID0gc2FsaW5pdHlfZ3JvdXAsIHkgPSBNTzIsIGZpbGwgPSBzYWxpbml0eV9ncm91cCksDQogICAgICAgICAgICAgIHNoYXBlID0gMjEsIHdpZHRoID0gMC4zLCBzaXplID0gMSwgY29sb3IgPSAiYmxhY2siLCBhbHBoYSA9IDAuMSkgKw0KICAgIGdlb21fcG9pbnQoZGF0YSA9IG1lYW5fbW8yX3NhbGluaXR5LA0KICAgICAgICAgICAgIGFlcyh4ID0gc2FsaW5pdHlfZ3JvdXAsIHkgPSBtZWFuX21vMiwgZmlsbCA9IHNhbGluaXR5X2dyb3VwKSwNCiAgICAgICAgICAgICBzaXplID0gNCwgYWxwaGEgPSAxLCBzdHJva2UgPSAyLCBjb2xvciA9ICJibGFjayIsIHNoYXBlID0gMjEsDQogICAgICAgICAgICAgIHBvc2l0aW9uID0gcG9zaXRpb25fbnVkZ2UoeCA9IDAuMDUpKSArDQogIHN0YXRfcG9pbnRpbnRlcnZhbChkYXRhID0gZW1tZWFuc19kcmF3cywgDQogICAgICAgICAgICAgICAgICAgICBhZXMoeCA9IHNhbGluaXR5X2dyb3VwLCB5ID0gLnZhbHVlKSwNCiAgICAgICAgICAgICAgICAgICAgIGNvbG9yID0gImJsYWNrIiwgZmlsbCA9ICJncmV5IiwgcG9pbnRfaW50ZXJ2YWwgPSAibWVhbl9xaSIsIC53aWR0aCA9IDAuOTUsIHNoYXBlID0gMjEsICBzdHJva2UgPSAyLCBwb2ludF9zaXplID0gNCwgYWxwaGEgPSAxLA0KICAgICAgICAgICAgICAgICAgICAgcG9zaXRpb24gPSBwb3NpdGlvbl9udWRnZSh4ID0gLTAuMDUpKSArDQogIHNjYWxlX2ZpbGxfbWFudWFsKHZhbHVlcyA9IGMoIiM0QjUzMjAiLCAiIzAwMDA4MCIpKSArICAjIEN1c3RvbSBmaWxsIGNvbG91cnMNCiAgc2NhbGVfY29sb3VyX21hbnVhbCh2YWx1ZXMgPSBjKCIjNEI1MzIwIiwgIiMwMDAwODAiKSkgKw0KICB0aGVtZV9jbGVhbigpICsNCiAgdGhlbWUobGVnZW5kLnBvc2l0aW9uID0gIm5vbmUiKSArDQogIGxhYnMoDQogICAgc3VidGl0bGUgPSAiIiwNCiAgICB4ID0gIlNhbGluaXR5IGdyb3VwIChwcHQpIiwNCiAgICB5ID0gIlJvdXRpbmUgTU8yIChtZyBPMiBnL2gpIg0KICApDQoNCmZpZ18xDQpgYGANCg0KKipGaWd1cmUgMWE6KiogUm91dGluZSBtZXRhYm9saWMgcmF0ZSAoaS5lLiBNTzxzdWI+Mjwvc3ViPiAobWdeLTFeIE88c3ViPjI8L3N1Yj4gaF4tMV4pIG1lYXN1cmVkIGR1cmluZyBTTVIgbWVhc3N1cm1lbnRzKSBwbG90dGVkIGJ5IHNhbGluaXR5IHRyZWF0bWVudC4gVGhlIHNtYWxsIHRyYW5zcGFyZW50IHBvaW50cyBhcmUgdGhlIG9ic2VydmVkIHZhbHVlcywgdGhlIHNoYWRlZCBhcmVhIGJlaGluZCB0aGUgcG9pbnRzIGlzIHRoZSBhIGtlcm5lbCBkZW5zaXR5IG9mIHRoZSBvYnNlcnZlZCBkYXRhLCB0aGUgbGFyZ2UgY29sb3VyZWQgcG9pbnQgKHRvIHRoZSByaWdodCkgaXMgdGhlIG9ic2VydmVkIG1lYW4sIHRoZSBsYXJnZSBncmV5IHBvaW50IHdpdGggZXJyb3IgYmFycyAodG8gdGhlIGxlZnQpIGlzIHRoZSBtb2RlbCBlc3RpbWF0ZWQgbWFyZ2luYWwgbWVhbiAoZWVtZWFuKSA5NSUgQ3JlZGlibGUgSW50ZXJ2YWxzICg5NSUgQ0kpLiANCg0KIyMgU01SDQoNCiMjIyBGb3JtYXRpbmcgYW5kIHNjYWxpbmcgZGF0YQ0KDQpIZXJlIHdlIGFyZSBmaWx0ZXJpbmcgdGhlIGRhdGEgZnJhbWUgdG8gaGF2ZSBvbmx5IG1lYXN1cmUgcGVyIGZpc2ggZm9yIHRoZSBTTVIgZXN0aW1hdGUgICANCg0KYGBge3J9DQpzY2FsZV9saXN0IDwtIGMoInRlbXBfbWVhbiIsICJtYXNzIiwgImN5Y2xlcyIpDQoNCnNtciA8LSBzbG9wZV90aWR5X3NtciAlPiUNCiAgZHBseXI6Omdyb3VwX2J5KGlkKSAlPiUgDQogIGRwbHlyOjpyZWZyYW1lKHRlbXBfbWVhbiA9IG1lYW4odGVtcCksDQogICAgICAgICAgICAgICAgIG1hc3MgPSBtYXNzWzFdLA0KICAgICAgICAgICAgICAgICBTTVIgPSBTTVJbMV0sDQogICAgICAgICAgICAgICAgIHNhbGluaXR5X2dyb3VwID0gc2FsaW5pdHlfZ3JvdXBbMV0sDQogICAgICAgICAgICAgICAgIGN5Y2xlcyA9IGxlbmd0aChvcmRlcikpICU+JSANCiAgZHBseXI6Om11dGF0ZShhY3Jvc3MoYWxsX29mKHNjYWxlX2xpc3QpLCB+IHNjYWxlKC54LCBjZW50ZXIgPSBUUlVFLCBzY2FsZSA9IEZBTFNFKSwNCiAgICAgICAgICAgICAgICAgICAgICAgLm5hbWVzID0gInsuY29sfV96IikpDQpgYGANCg0KIyMjIE1vZGVsIHN0cnVjdHVyZQ0KDQpIZXJlIHdlIHdpbGwgdXNlIGEgQmF5ZXNpYW4gR2VuZXJhbGlzZWQgTGluZWFyIE1peGVkIE1vZGVsIChHTE1NKSB3aXRoIGEgR2FtbWEgZGlzdHJpYnV0aW9uIGFuZCBhIGxvZyBsaW5rIGBHYW1tYShsaW5rID0gImxvZyIpYCwgd2hlcmUgdGhlIHNoYXBlIHBhcmFtZXRlciAoSykgaXMgYWxzbyBtb2RlbGxlZCBhcyBhIGZ1bmN0aW9uIG9mIHRoZSBzYWxpbml0eSBncm91cCwgYHNoYXBlIH4gc2FsaW5pdHlfZ3JvdXBgLiBUaGlzIG1vZGVscyBNTzIgYnkgc2FsaW5pdHkgZHVyaW5nIHRoZSBTTVIgcGhhc2UgdG8gc2VlIGlmIHRoZSBmaXNoIGhlbGQgYXQgZGlmZmVyZW50IHNhbGluaXRpZXMgaGF2ZSBkaWZmZXJlbnQgU01Scy4gV2UgaGF2ZSBhbHNvIGFkZGVkIGEgZmV3IHNjYWxlZCBwcmVkaWN0b3JzLCB0aGF0IG1heSBoZWxwIGRlc2NyaWJlIHZhcmlhdGlvbiBpbiB0aGUgZGF0YSwgc3VjaCBhcyBtYXNzIChnOyAwLjIx4oCTMS42KSB0ZW1wZXJhdHVyZSAowrBDOyAxMy44NDHigJMxNC4yNzcpLCBtZWFzdXJlbWVudCBvcmRlciAoMeKAkzI4KSwgYW5kIGxpZ2h0L2RhcmsgY3ljbGUgKGxpZ2h0IG9yIGRhcms7IGxpZ2h0IGJldHdlZW4gMDc6MDA6MDAgYW5kIDE5OjAwOjAwKSwgd2UgYWxzbyBpbmNsdWRlIGEgcmFuZG9tIGVmZmVjdCBmb3IgZmlzaCBpZCB0byBhY2NvdW50IGZvciBtdWx0aXBsZSBNTzIgbWVhc3VyZXMgb24gZWFjaCBmaXNoICgxIC0gNTgpLiBXZSBhbGxvd2VkIHRoZSB0aGUgc2hhcGUgcGFyYW1ldGVyIChLKSB0byB2YXJ5IGFzIGEgZnVuY3Rpb24gb2Ygc29tZSBvZiB0aGUgcHJlZGljdG9ycyAoZS5nLiBzYWxpbml0eV9ncm91cCwgb3JkZXJfeikgdG8gaW1wcm92ZSBmaXQuICAgDQoNCmBgYHtyfQ0Kc21yX2dhbW1hX2JmIDwtIGJmKFNNUiB+IHRlbXBfbWVhbl96ICsgDQogICAgICAgICAgICAgICAgICBjeWNsZXNfeiAgKw0KICAgICAgICAgICAgICAgICAgbWFzc196ICsgDQogICAgICAgICAgICAgICAgICBzYWxpbml0eV9ncm91cCwNCiAgICAgICAgICAgICAgICAgIHNoYXBlIH4gc2FsaW5pdHlfZ3JvdXAsDQogICAgICAgICAgICAgICAgICBmYW1pbHkgPSBHYW1tYShsaW5rID0gImxvZyIpKQ0KYGBgDQoNCg0KIyMjIFByaW9yIHNlbGVjdGlvbg0KDQpUaGVzZSBhcmUgdGhlIGRlZmF1bHQgcHJpb3JzIGZvciB0aGUgbW9kZWwuIFdlIHdpbGwgdXNlIHRoZXNlLg0KDQpgYGB7cn0NCnByaW9yc19kZWZhdWx0IDwtIGdldF9wcmlvcihzbXJfZ2FtbWFfYmYsIGRhdGEgPSBzbXIsIGZhbWlseSA9IEdhbW1hKGxpbmsgPSAibG9nIikpDQpwcmlvcnNfZGVmYXVsdCAlPiUgDQogIGd0KCkNCmBgYA0KDQojIyMgUnVuIG1vZGVsDQoNCkhlcmUgd2UgcnVuIHRoZSBtb2RlbCwgSSBoYXZlIGhhc2hlZCB0aGlzIG91dCBiZWNhdXNlIEkgaGF2ZSBzYXZlZCB0aGUgbW9kZWwgZm9yIHF1aWNrIHJlbG9hZGluZy4gIA0KDQrij63vuI8gKipTa2lwIHRoaXMgc3RlcCoqIGlmIHlvdSBoYXZlIGRvd25sb2FkZWQgdGhlIHNhdmVkIG1vZGVsICdzbXJfbW9kX2dhbW1hLnJkcycNCg0KYGBge3IsIGVjaG89RkFMU0V9DQojIHNldHdkKG91dHB1dF9tb2RzX3dkKQ0KIyANCiMgc21yX21vZF9nYW1tYSA8LSBicm0oc21yX2dhbW1hX2JmLA0KIyAgICAgICAgICAgICAgICBkYXRhID0gc21yLA0KIyAgICAgICAgICAgICAgICBjb3JlcyA9IDQsDQojICAgICAgICAgICAgICAgIGNoYWlucyA9IDQsDQojICAgICAgICAgICAgICAgIHByaW9yID0gcHJpb3JzX2RlZmF1bHQsDQojICAgICAgICAgICAgICAgIHdhcm11cCA9IDEwMDAsDQojICAgICAgICAgICAgICAgIHNlZWQgPSAxNDMwMTksDQojICAgICAgICAgICAgICAgIHRoaW4gPSAyLA0KIyAgICAgICAgICAgICAgICBpdGVyID0gODAwMCwNCiMgICAgICAgICAgICAgICAgc2F2ZV9wYXJzID0gc2F2ZV9wYXJzKGFsbD1UUlVFKSwNCiMgICAgICAgICAgICAgICAgc2FtcGxlX3ByaW9yID0gVFJVRSwNCiMgICAgICAgICAgICAgICAgZmlsZSA9ICdzbXJfbW9kX2dhbW1hJykNCiMgcHJpbnQoIk1vZGVsIGNvbXBsZXRlIikNCmBgYA0KDQpIZXJlIHdlIHJlbG9hZCB0aGUgbW9kZWwgICANCg0KYGBge3J9DQpzZXR3ZChvdXRwdXRfbW9kc193ZCkNCg0Kc21yX21vZF9nYW1tYSA8LSAgcmVhZFJEUyhmaWxlID0gInNtcl9tb2RfZ2FtbWEucmRzIikNCmBgYA0KDQojIyMgTW9kZWwgZGlhZ25vc3RpY3MNCg0KQ2hlY2tpbmcgbW9kZWwgY29udmVyZ2VuY2UgICANCg0KYGBge3J9DQpjb2xvcl9zY2hlbWVfc2V0KCJyZWQiKQ0KcGxvdChzbXJfbW9kX2dhbW1hLCBhc2sgPSBGKQ0KYGBgDQoNClVzaW5nIGxlYXZlIG9uZSBvdXQgKGxvbykgbWVhc3VyZSBvZiBmaXQsIHRoZSBtb2RlbCBhcHBlYXJzIHRvIHByZWZvcm0gd2VsbCwgb25lIFBhcmV0byBrIGVzdGltYXRlcyBmYWxscyBvdXRzaWRlIHRoZSBnb29kIHJhbmdlICgwLjcsIDFdICAgDQoNCmBgYHtyfQ0KbG9vKHNtcl9tb2RfZ2FtbWEpDQpgYGANCg0KTW9kZWwgcHJlZGljdGlvbnMgZ2VuZXJhbGx5IGFsaWduIHdpdGggdGhlIG9ic2VydmVkIGRhdGEsIGJ1dCB0aGVyZSBpcyBhIGxvdCBvZiB1bmNlcnRhaW50eS4gICAgICAgDQoNCg0KYGBge3J9DQpjb2xvcl9zY2hlbWVfc2V0KCJyZWQiKQ0KcCA8LSBwcF9jaGVjayhzbXJfbW9kX2dhbW1hLCB0eXBlID0gImRlbnNfb3ZlcmxheSIpDQpwDQpgYGANCg0KDQojIyMg8J+TiCBSZXN1bHRzDQoNCldlIGRpZCBub3Qgc2VlIGEgbWVhbmluZ2Z1bCBkaWZmZXJlbmNlIGJldHdlZW4gdGhlIHJvdXRpbmUgbWV0YWJvbGljIHJhdGUgZm9yIGZpc2ggZnJvbSB0aGUgdHdvIHNhbGluaXR5IHRyZWF0bWVudHMuIA0KDQojIyMjIFRhYmxlIFMyIA0KDQoqKlRhYmxlIFMyOioqIEZpeGVkIGVmZmVjdCBFc3RpbWF0ZXMgKM6yKSBhbmQgOTUlIENyZWRpYmxlIEludGVydmFscyAoOTUlIENJKSBmcm9tIGEgQmF5ZXNpYW4gR2VuZXJhbGlzZWQgTGluZWFyIE1peGVkIE1vZGVsIChHTE1NKSB3aXRoIGEgR2FtbWEgZGlzdHJpYnV0aW9uIGFuZCBhIGxvZyBsaW5rDQoNCmBgYHtyfQ0KbW9kZWxfZXN0IDwtIGZpeGVmKHNtcl9tb2RfZ2FtbWEsIHByb2JzID0gYygwLjAyNSwgMC45NzUpKSAlPiUgDQogIGFzLmRhdGEuZnJhbWUoKSAlPiUgDQogIHRpYmJsZTo6cm93bmFtZXNfdG9fY29sdW1uKHZhciA9ICJQcmVkaWN0b3IiKSAlPiUgDQogIGRwbHlyOjptdXRhdGUoJ86yJyA9IHJvdW5kKEVzdGltYXRlLCAzKSwNCiAgICAgICAgICAgICAgICBRMi41ID0gcm91bmQoUTIuNSwgMyksDQogICAgICAgICAgICAgICAgUTk3LjUgPSByb3VuZChROTcuNSwgMyksDQogICAgICAgICAgICAgICAgJzk1JSBDSScgPSBwYXN0ZTAoIlsiLCBRMi41LCAiLCAiLCBROTcuNSwgIl0iKSkNCg0KbW9kZWxfZXN0ICU+JSANCiAgZHBseXI6OnNlbGVjdChQcmVkaWN0b3IsICfOsicsICc5NSUgQ0knKSAlPiUgDQogIGd0KCkNCmBgYA0KICAgICAgDQogIA0KTG9va2luZyBhdCB0aGUgbWFyZ2luYWwgbWVhbiBkaWZmZXJlbmNlIGJldHdlZW4gc2FsaW5pdHkgZ3JvdXBzICAgDQoNCmBgYHtyfQ0KZW1fcmVzdWx0cyA8LSBlbW1lYW5zKHNtcl9tb2RfZ2FtbWEsIH4gc2FsaW5pdHlfZ3JvdXApIA0KY29udHJhc3RfcmVzdWx0cyA8LSBjb250cmFzdChlbV9yZXN1bHRzLCBtZXRob2QgPSAicGFpcndpc2UiKQ0KZW1fcmVzdWx0c19kZiA8LSAgZW1fcmVzdWx0cyAlPiUgdGlkeSgpICU+JSANCiAgbXV0YXRlKGFjcm9zcyh3aGVyZShpcy5udW1lcmljKSwgfiBleHAoLikpKQ0KY29udHJhc3RfcmVzdWx0c19kZiA8LSAgY29udHJhc3RfcmVzdWx0cyAlPiUgdGlkeSgpICU+JSANCiAgbXV0YXRlKGFjcm9zcyh3aGVyZShpcy5udW1lcmljKSwgfiBleHAoLikpKQ0KDQoNCmVtX3Jlc3VsdHNfZGYgJT4lIA0KICBndCgpDQpgYGANCiAgDQpQdWxsaW5nIHRoZSBlbW1lYW5zIGRyYXdzIGZvciBvdXIgcGxvdCAgIA0KDQpgYGB7cn0NCmVtbWVhbnNfZHJhd3MgPC0gc21yX21vZF9nYW1tYSAlPiUNCiAgZW1tZWFucyh+IHNhbGluaXR5X2dyb3VwKSAlPiUNCiAgZ2F0aGVyX2VtbWVhbnNfZHJhd3MoKSAlPiUgDQogIGRwbHlyOjptdXRhdGUoLnZhbHVlID0gIGV4cCgudmFsdWUpLA0KICAgICAgICAgICAgICAgIHNhbGluaXR5X2dyb3VwID0gYXMuY2hhcmFjdGVyKHNhbGluaXR5X2dyb3VwKSkNCg0KZW1tZWFuc19jb250cmFzdF9kcmF3cyA8LSAgc21yX21vZF9nYW1tYSAlPiUNCiAgZW1tZWFucyh+IHNhbGluaXR5X2dyb3VwKSAlPiUNCiAgY29udHJhc3QobWV0aG9kID0gInBhaXJ3aXNlIikgJT4lDQogIGdhdGhlcl9lbW1lYW5zX2RyYXdzKCkgJT4lIA0KICBkcGx5cjo6bXV0YXRlKC52YWx1ZSA9ICBleHAoLnZhbHVlKSkNCmBgYA0KDQojIyMjIEZpZ3VyZSBTOQ0KDQpUaGlzIHBsb3QgaXMgaW4gdGhlIG1haW4gdGV4dCBvZiB0aGUgbWFudXNjcmlwdCBhcyAqKkZpZ3VyZSAxYioqDQoNCmBgYHtyfQ0KbWVhbl9zbXJfc2FsaW5pdHkgPC0gc21yICU+JQ0KICBkcGx5cjo6Z3JvdXBfYnkoc2FsaW5pdHlfZ3JvdXApICU+JQ0KICBkcGx5cjo6cmVmcmFtZShtZWFuX1NNUiA9IG1lYW4oU01SLCBuYS5ybSA9IFRSVUUpKQ0KDQoNCmZpZ18xYiA8LSBnZ3Bsb3QoKSArDQogIGdlb21fdmlvbGluKGRhdGEgPSBzbXIsDQogICAgICAgICAgICAgIGFlcyh4ID0gc2FsaW5pdHlfZ3JvdXAsIHkgPSBTTVIsIGZpbGwgPSBzYWxpbml0eV9ncm91cCksDQogICAgICAgICAgICAgIGNvbG9yID0gTkEsIGFscGhhID0gMC4yKSArDQogIGdlb21faml0dGVyKGRhdGEgPSBzbXIsDQogICAgICAgICAgICAgIGFlcyh4ID0gc2FsaW5pdHlfZ3JvdXAsIHkgPSBTTVIsIGZpbGwgPSBzYWxpbml0eV9ncm91cCksDQogICAgICAgICAgICAgIHNoYXBlID0gMjEsIHdpZHRoID0gMC4zLCBzaXplID0gMSwgY29sb3IgPSAiYmxhY2siLCBhbHBoYSA9IDAuMSkgKw0KICAgIGdlb21fcG9pbnQoZGF0YSA9IG1lYW5fc21yX3NhbGluaXR5LA0KICAgICAgICAgICAgIGFlcyh4ID0gc2FsaW5pdHlfZ3JvdXAsIHkgPSBtZWFuX1NNUiwgZmlsbCA9IHNhbGluaXR5X2dyb3VwKSwNCiAgICAgICAgICAgICBzaXplID0gNCwgYWxwaGEgPSAxLCBzdHJva2UgPSAyLCBjb2xvciA9ICJibGFjayIsIHNoYXBlID0gMjEsDQogICAgICAgICAgICAgIHBvc2l0aW9uID0gcG9zaXRpb25fbnVkZ2UoeCA9IDAuMDUpKSArDQogIHN0YXRfcG9pbnRpbnRlcnZhbChkYXRhID0gZW1tZWFuc19kcmF3cywNCiAgICAgICAgICAgICAgICAgICAgIGFlcyh4ID0gc2FsaW5pdHlfZ3JvdXAsIHkgPSAudmFsdWUpLA0KICAgICAgICAgICAgICAgICAgICAgY29sb3IgPSAiYmxhY2siLCBmaWxsID0gImdyZXkiLCBwb2ludF9pbnRlcnZhbCA9ICJtZWFuX3FpIiwgLndpZHRoID0gMC45NSwgc2hhcGUgPSAyMSwgIHN0cm9rZSA9IDIsIHBvaW50X3NpemUgPSA0LCBhbHBoYSA9IDEsDQogICAgICAgICAgICAgICAgICAgICBwb3NpdGlvbiA9IHBvc2l0aW9uX251ZGdlKHggPSAtMC4wNSkpICsNCiAgc2NhbGVfZmlsbF9tYW51YWwodmFsdWVzID0gYygiIzRCNTMyMCIsICIjMDAwMDgwIikpICsgICMgQ3VzdG9tIGZpbGwgY29sb3Vycw0KICBzY2FsZV9jb2xvdXJfbWFudWFsKHZhbHVlcyA9IGMoIiM0QjUzMjAiLCAiIzAwMDA4MCIpKSArDQogIHRoZW1lX2NsZWFuKCkgKw0KICB0aGVtZShsZWdlbmQucG9zaXRpb24gPSAibm9uZSIpICsNCiAgbGFicygNCiAgICBzdWJ0aXRsZSA9ICIiLA0KICAgIHggPSAiU2FsaW5pdHkgZ3JvdXAgKHBwdCkiLA0KICAgIHkgPSAiU3RhbmRhcmQgbWV0YWJvbGljIHJhdGUgKFNNUjsgbWcgTzIgZy9oKSINCiAgKQ0KDQpmaWdfMWINCmBgYA0KDQoqKkZpZ3VyZSAxYjoqKiBUaGUgc3RhbmRhcmQgbWV0YWJvbGljIHJhdGUgZXN0aW1hdGUgKG1nXi0xXiBPPHN1Yj4yPC9zdWI+IGheLTFeKSBwbG90dGVkIGJ5IHNhbGluaXR5IHRyZWF0bWVudC4gVGhlIHNtYWxsIHRyYW5zcGFyZW50IHBvaW50cyBhcmUgdGhlIG9ic2VydmVkIHZhbHVlcywgdGhlIHNoYWRlZCBhcmVhIGJlaGluZCB0aGUgcG9pbnRzIGlzIHRoZSBhIGtlcm5lbCBkZW5zaXR5IG9mIHRoZSBvYnNlcnZlZCBkYXRhLCB0aGUgbGFyZ2UgY29sb3VyZWQgcG9pbnQgKHRvIHRoZSByaWdodCkgaXMgdGhlIG9ic2VydmVkIG1lYW4sIHRoZSBsYXJnZSBncmV5IHBvaW50IHdpdGggZXJyb3IgYmFycyAodG8gdGhlIGxlZnQpIGlzIHRoZSBtb2RlbCBlc3RpbWF0ZWQgbWFyZ2luYWwgbWVhbiAoZWVtZWFuKSA5NSUgQ3JlZGlibGUgSW50ZXJ2YWxzICg5NSUgQ0kpLiAgDQoNCg0KIyMgSW5jcmVtZW50YWwgcmVncmVzc2lvbiBhbmFseXNlcw0KDQpIZXJlIHdlIGFyZSBmb2xsb3dpbmcgdGhlIG1ldGhvZHMgVXJiaW5hIGV0IGFsLiAoMjAxMileWzFdXiB3aXRoIGFuIGluY3JlbWVudGFsIHJlZ3Jlc3Npb24gYW5hbHlzZXMsIGluIG9yZGVyIHRvIGRldGVybWluZSB0aGUgYmVzdCBmaXQgZm9yIHRoZSBNTzxzdWI+Mjwvc3ViPiB2cyBPPHN1Yj4yPC9zdWI+IGRhdGEgICANCg0KVGhpcyBhbmFseXNpcyBhcHByb2FjaCBldmFsdWF0ZXMgdGhlIHJlbGF0aXZlICdmaXQnIG9mIGVhY2ggcG9seW5vbWlhbCBvcmRlciBlcXVhdGlvbiBzdGFydGluZyBhdCB6ZXJvIGFuZCBpbmNyZWFzaW5nIHRvIHRoZSB0aGlyZCBvcmRlciwgcGVybWl0dGluZyBhIG1hdGhlbWF0aWNhbCBhc3Nlc3NtZW50IG9mIHdoZXRoZXIgZmlzaCB3ZXJlIG94eWNvbmZvcm1pbmcgb3Igb3h5cmVndWxhdG9yaW5nLiBJZiB0aGUgZGF0YSBpcyBiZXN0IGZpdHRlZC9wcmVkaWN0ZWQgYnkgYSBzaW5nbGUgbGluZWFyIHJlbGF0aW9uc2hpcCAoMV5zdF4tb3JkZXIgcG9seW5vbWlhbCkgd2l0aCBhIHBvc2l0aXZlIHNsb3BlLCB0aGlzIHdvdWxkIHN1Z2dlc3RzIHRoZSBmaXNoIHdlcmUgb3h5Y29uZm9ybWluZy4gQWx0ZXJuYXRlbHksIGlmIHRoZSByZWxhdGlvbnNoaXAgaXMgYmVzdCBtb2RlbGxlZCBieSBhIGZsYXQgcmVncmVzc2lvbiAoMF50aF4tb3JkZXIgcG9seW5vbWlhbCksIG9yIGEgaGlnaGVyIG9yZGVyIHBvbHlub21pYWwgKDJebmReIG9yIDNecmReLW9yZGVyIHBvbHlub21pYWwpIHRoZSBmaXNoIGlzIGxpa2VseSBveHlyZWd1bGF0b3JpbmcuICAgDQoNCmBgYHtyfQ0Kb3V0cHV0X21vZHNfYmF5ZXNfd2QgPC0gcGFzdGUwKG91dHB1dF9tb2RzX3dkLCAiLi9iYXllcy1yZWdzIikNCmlmICghZGlyLmV4aXN0cyhvdXRwdXRfbW9kc19iYXllc193ZCkpIHtkaXIuY3JlYXRlKG91dHB1dF9tb2RzX2JheWVzX3dkKX0NCmBgYA0KDQojIyMgQnVpbGRpbmcgQmF5ZXNpYW4gcmVncmVzc2lvbnMNCg0KSGVyZSB3ZSBhcmUgdXNpbmcgYSBCYXllc2lhbiBhcHByb2FjaCB0byBtb2RlbCBmaXR0aW5nIHdpdGggYnJtLiBUaGVzZSBtb2RlbHMgdGFrZSBhIGxvbmcgdGltZSB0byBydW4sIHNvIEkgaGF2ZSBzYXZlZCB0aGVtIGFuZCByZS1sb2FkZWQgdGhlbSB0byBzYXZlIHRpbWUuIEkgaGF2ZSBhbHNvIHNhdmVkIHRoZSBzdW1tYXJ5IGRhdGEgcHJvZHVjZWQgZnJvbSB0aGUgbW9kZWxzLCB0byBzYXZlIHRpbWUsIHlvdSBjYW4gc2ltcGx5IHNraXAgdGhlIGhhc2hlZCBjb2RlIGFuZCBpbnB1dCB0aGUgcmVzdWx0aW5nIHN1bW1hcnkgZGF0YS4gICANCg0KV2Ugd2lsbCBydW4gb3VyIGN1c3RvbSBmdW5jdGlvbiwgYGJheWVzX2luY3JlbWVudGFsX3JlZ3Jlc3Npb25fYnlfaWRgLiAgIA0KDQrij63vuI8gKipTa2lwIHRoaXMgc3RlcCoqIGlmIHlvdSBoYXZlIGFscmVhZHkgcnVuIHRoaXMgb25jZSwgb3IgaGF2ZSBkb3dubG9hZGVkIHRoZSBzYXZlZCBtb2RlbHMgb3Igc2F2ZWQgZGF0YWZpbGVzIGZyb20gR2l0SHViICh0aGF0J3Mgd2h5IGl0cyBoYXNoZWQgb3V0KS4gKlRoaXMgY29kZSB0YWtlcyBhIHdoaWxlIHRvIHJ1biogIA0KDQpJIGhhdmUgZnVuZCB0aGF0IEkgZ2V0IGFuIGVycm9yLCAnQ2F1c2VkIGJ5IGVycm9yIGluIGBzb2NrZXRDb25uZWN0aW9uKClgOicuIEkgdGhpbmsgdGhlIHN5c3RlbSBtYXkgYmUgaGl0dGluZyBhIGxpbWl0IG9uIHRoZSBudW1iZXIgb2Ygc2ltdWx0YW5lb3VzIHNvY2tldCBjb25uZWN0aW9ucy4gSWYgeW91IHRyeSBhbmQgcnVuIHRoaXMgY29kZSBhbmQgYWxzbyBnZXQgdGhpcyBpc3N1ZSwgdHJ5IHJlZHVjaW5nIHRoZSBudW1iZXIgb2YgcGFyYWxsZWwgd29ya2VyczogcGxhbihtdWx0aXNlc3Npb24sIHdvcmtlcnMgPSAyKS4gICAgDQoNCmBgYHtyfQ0KIyBpZHMgPC0gc2xvcGVfdGlkeSAlPiUNCiMgICBkcGx5cjo6ZGlzdGluY3QoaWQpICU+JQ0KIyAgIHB1bGwoaWQpDQojIA0KIyBwbGFuKG11bHRpc2Vzc2lvbikNCiMgDQojIGZ1dHVyZV9tYXAoDQojICAgaWRzLA0KIyAgIGJheWVzX2luY3JlbWVudGFsX3JlZ3Jlc3Npb25fYnlfaWQsDQojICAgaWRfbmFtZSA9ICJpZCIsDQojICAgZGF0YSA9IHNsb3BlX3RpZHksDQojICAgcmVzcG9uc2UgPSAiTU8yX2ciLA0KIyAgIHByZWRpY3RvciA9ICJETyIsDQojICAgc2VlZF9udW1iZXIgPSAxNDMwMTksDQojICAgc2F2ZV9tb2RlbHMgPSBUUlVFLA0KIyAgIG1vZF9vdXRwdXRfd2QgPSBvdXRwdXRfbW9kc19iYXllc193ZA0KIyApDQojIA0KIyBwbGFuKHNlcXVlbnRpYWwpDQpgYGANCg0KTG9hZCBhbGwgbW9kZWxzIGFuZCBzdG9yZSBpbiBhIGxpc3QsIHdpbGwgdXNlIGEgbG90IG9mIG1lbW9yeS4gWW91IGNhbiBhbHNvIHNraXAgdGhpcyBzdGVwIGFuZCBsb2FkIHRoZSByZXN1bHRpbmcgZGF0YSBmcmFtZXMgYmVsb3cuIEkgYW0gdXNpbmcgdGhlIGN1c3RvbSBmdW5jdGlvbiBgbG9hZF9yZHNgLCBzbyB3ZSBjYW4gY29tcGFyZSB0aGVtIGFuZCBnZW5lcmF0ZSBwcmVkaWN0aW9ucy4gIA0KDQrij63vuI8gKipTa2lwIHRoaXMgc3RlcCoqIGlmIHlvdSBoYXZlIGRvd25sb2FkZWQgdGhlIHNhdmVkIGRhdGEgcHVsbGVkIGZyb20gdGhlc2UgbW9kZWxzIGhhdmUgYWxzbyBoYXNlZCB0aGlzIG91dCwgYmF5ZXNfcmVnX21vZHNfZml0LmNzdicgYW5kICdiYXllc19yZWdfbW9kc19wcmVkaWN0aW9ucy5jc3YnLiBUaGVzZSBhcmUgaW4gdGhlIG1vZC1kYXRhIGRpcmVjdG9yeS4gDQoNCmBgYHtyfQ0KI2JheWVzX3JlZ19tb2RzIDwtIGxvYWRfcmRzKG1vZGVsX2R3ID0gb3V0cHV0X21vZHNfYmF5ZXNfd2QpDQpgYGANCg0KIyMjIE1vZGVsIGZpdHMNCg0K4o+t77iPICoqU2tpcCB0aGlzIHN0ZXAqKiBpZiB5b3UgaGF2ZSBkb3dubG9hZGVkICoq8J+SvyBiYXllc19yZWdfbW9kc19maXQuY3N2KiouICAgIA0KDQpJdCBnZXRzIG1vZGVsIGZpdCBwYXJhbWV0ZXJzIGxvbyBhbmQgcjIgdXNpbmcgdGhlIGN1c3RvbSBmdW5jdGlvbiwgYGluY3JlbWVudGFsX3JlZ3Jlc3Npb25fYmF5ZXNfZml0c2AuICpUaGlzIGNvZGUgdGFrZXMgYSB3aGlsZSB0byBydW4qICAgDQoNCmBgYHtyfQ0KIyBzZXR3ZChtb2RfZGF0YV93ZCkNCiMgYmF5ZXNfcmVnX21vZHNfZml0IDwtIGluY3JlbWVudGFsX3JlZ3Jlc3Npb25fYmF5ZXNfZml0cyhtb2RlbHMgPSBiYXllc19yZWdfbW9kcykNCiMgd3JpdGUuY3N2KGJheWVzX3JlZ19tb2RzX2ZpdCwgImJheWVzX3JlZ19tb2RzX2ZpdC5jc3YiLCByb3cubmFtZXMgPSBGQUxTRSkNCmBgYA0KICANClJlYWRpbmcgaW4gdGhpcyBtb2RlbCBmaXQgZGF0YSBmcmFtZSwgaW4gdGhlIGNhc2UgeW91IGRpZCBub3QgbG9hZCBpbiBhbGwgdGhlIG1vZGVscy4gICAgDQoNCmBgYHtyfQ0Kc2V0d2QobW9kX2RhdGFfd2QpDQpiYXllc19yZWdfbW9kc19maXQgPC0gcmVhZC5jc3YoImJheWVzX3JlZ19tb2RzX2ZpdC5jc3YiKQ0KYGBgDQoNCioqU2VsZWN0aW5nIHRoZSBiZXN0IGZpdHRpbmcgbW9kZWwqKiAgDQogIA0KZWxwZF9sb28sIG9yIHRoZSBleHBlY3RlZCBsb2cgcG9pbnR3aXNlIHByZWRpY3RpdmUgZGVuc2l0eSBmb3IgbGVhdmUtb25lLW91dCBjcm9zcy12YWxpZGF0aW9uLCBpcyBhIG1ldHJpYyB1c2VkIGluIEJheWVzaWFuIG1vZGVsIGV2YWx1YXRpb24gdG8gYXNzZXNzIHRoZSBwcmVkaWN0aXZlIGFjY3VyYWN5IG9mIGEgbW9kZWwuIFRoZSBlbHBkX2xvbyBpcyBhbiBhcHByb3hpbWF0aW9uIG9mIGhvdyB3ZWxsIHRoZSBtb2RlbCBpcyBleHBlY3RlZCB0byBwcmVkaWN0IG5ldyBkYXRhLCBiYXNlZCBvbiBsZWF2ZS1vbmUtb3V0IGNyb3NzLXZhbGlkYXRpb24uIEhpZ2hlciBlbHBkX2xvbyB2YWx1ZXMgaW5kaWNhdGUgYmV0dGVyIHByZWRpY3RpdmUgcGVyZm9ybWFuY2UuICAgDQoNCmBgYHtyfQ0KYmVzdF9maXRfYmF5ZXNfcmVnIDwtIGJheWVzX3JlZ19tb2RzX2ZpdCAlPiUNCiAgZHBseXI6Omdyb3VwX2J5KGlkKSAlPiUNCiAgZHBseXI6Om11dGF0ZShlbHBkX2xvb19yYW5rID0gcmFuaygtZWxwZF9sb28pKSAlPiUNCiAgZHBseXI6OnNlbGVjdChpZCwgbW9kZWxfdHlwZSwgZWxwZF9sb28sIHIyLCBlbHBkX2xvb19yYW5rLCByMl9xMi41LCByMl9xOTcuNSwgZXN0aW1hdGVfRE8sIGNvbmYubG93X0RPLCBjb25mLmhpZ2hfRE8pICU+JQ0KICB1bmdyb3VwKCkNCmBgYA0KDQoNCiMjIyBNb2RlbCBwcmVkaWN0aW9ucw0KDQrij63vuI8gKipTa2lwIHRoaXMgc3RlcCoqIGlmIHlvdSBoYXZlIGRvd25sb2FkZWQgKirwn5K/IGJheWVzX3JlZ19tb2RzX3ByZWRpY3Rpb25zLmNzdioqLiAgICANCg0KSXQgcHVsbHMgb3VyIG1vZGVsIHByZWRpY3Rpb25zIHVzaW5nIGEgY3VzdG9tIGZ1bmN0aW9uIGBiYXllc19tb2RfcHJlZGljdGlvbnNgLiAgIA0KDQpgYGB7cn0NCiMgc2V0d2QobW9kX2RhdGFfd2QpDQojIGJheWVzX3JlZ19tb2RzX3ByZWRpY3Rpb25zIDwtIGJheWVzX21vZF9wcmVkaWN0aW9ucyhtb2RlbHMgPSBiYXllc19yZWdfbW9kcywgb3JpZ2luYWxfZGF0YSA9IHNsb3BlX3RpZHkpDQojIHdyaXRlLmNzdihiYXllc19yZWdfbW9kc19wcmVkaWN0aW9ucywgImJheWVzX3JlZ19tb2RzX3ByZWRpY3Rpb25zLmNzdiIsIHJvdy5uYW1lcyA9IEZBTFNFKQ0KYGBgDQoNClJlYWRpbmcgaW4gdGhlIHByZWRpY3RlZCBkYXRhICAgDQoNCmBgYHtyfQ0Kc2V0d2QobW9kX2RhdGFfd2QpDQpiYXllc19yZWdfbW9kc19wcmVkaWN0aW9ucyA8LSByZWFkLmNzdigiYmF5ZXNfcmVnX21vZHNfcHJlZGljdGlvbnMuY3N2IikNCmBgYA0KDQpXZSBhcmUgZ29pbmcgdG8gY29tYmluZWQgdGhpcyB3aXRoIG91ciBiZXN0IGZpdHRpbmcgbW9kZWwgZGYsIHNvIHdlIGtub3cgaG93IHRoZXkgcmFua3MgZm9yIExPTy4gICANCg0KYGBge3J9DQpiYXllc19yZWdfbW9kc19wcmVkaWN0aW9ucyA8LSBsZWZ0X2pvaW4oYmF5ZXNfcmVnX21vZHNfcHJlZGljdGlvbnMsIGJlc3RfZml0X2JheWVzX3JlZywgYnkgPSBjKCJpZCIsICJtb2RlbF90eXBlIikpDQpgYGANCg0KIyMjIPCfk4ggUmVzdWx0cw0KDQojIyMjIE1vZGVsIHNlbGVjdGlvbiBzdW1tYXJ5DQoNClRoZSBiZXN0IGZpdHRpbmcgbW9kZWxzIHdhcyBtb3N0IG9mdGVuIGEgMl5uZF4tb3JkZXIgcG9seW5vbWlhbCAoKm4qID0gMjUsIDQzJSkuIFRoaXMgY291bGQgc3VnZ2VzdCB0aGUgcHJlc2VuY2Ugb2YgYSBjcml0aWNhbCBveHlnZW4gdGhyZXNob2xkIChQY3JpdCkgd2hlcmUgdGhlIHJlbGF0aW9uc2hpcCBiZXR3ZWVuIE88c3ViPjI8L3N1Yj4gYW5kIE1PPHN1Yj4yPC9zdWI+IGNoYW5nZXMuIFRvIGNvbmZpcm0gdGhlaXIgaXMgYSBQY3JpdCwgd2UgbmVlZCB0byB2YWxpZGF0ZWQgdGhlIHNoYXBlIG9mIHRoZSBwb2x5bm9taWFscyBhbmQgaW4gc2hvdWxkIHVzZSBhIG1vcmUgc3BlY2lmaWMgbW9kZWwgdG8gdGVzdCB0aGUgUGNyaXQgdmFsdWUuIEluIGFueSBjYXNlLCBUaGlzIHR5cGUgb2YgbW9kZWwgaXMgaW5kaWNhdGl2ZSBvZiAqKm94eXJlZ3VsYXRvcioqLiBUaGUgc2FtZSBpcyBhbHNvIHRydWUgZm9yIHRob3NlIHdpdGggYSAzXnJkXi1vcmRlciBwb2x5bm9taWFsICgqbiogPSAxMiwgMjElKS4gICAgIA0KDQpUaGUgbmV4dCBtb3N0IGNvbW1vbiBpcyAxXnN0Xi1vcmRlciBwb2x5bm9taWFsICgqbiogPSAxOCwgMzElKS4gSW4gdGhlIGNhc2Ugb2YgdGhlIDFec3ReLW9yZGVyIHBvbHlub21pYWxzLCBpdCBzdWdnZXN0IHRoZSBwcmVzZW5jZXMgb2YgbGluZWFyIHJlbGF0aW9uc2hpcCBiZXR3ZWVuIG8yIGFuZCBNTzIsIHdoaWNoIGlzIGluZGljYXRpdmUgb2YgKipveHljb25mb3JtZXIqKi4gSG93ZXZlciwgdG8gYmUgdHJ1ZSBldmlkZW5jZSBvZiBhIG94eWNvbmZvcm1lciB0aGlzIHJlbGF0aW9uc2hpcCBzaG91bGQgYmUgcG9zaXRpdmUgKGkuZS4gYXMgTzxzdWI+Mjwvc3ViPiBmYWxscyBNTzxzdWI+Mjwvc3ViPiBhbHNvIGZhbGxzKS4gMTMgb2YgdGhlIDE4IGluZGl2aWR1YWxzIGJlc3QgbW9kZWxsZWQgd2l0aCBhIGxpbmVhciBmdW5jdGlvbiBoYWQgcG9zaXRpdmUgZXN0aW1hdGVzIHdpdGggY3JlZGlibGUgaW50ZXJ2YWxzIHRoYXQgZGlkIG5vdCBvdmVybGFwIHdpdGggemVybyAoVGFibGUgUzMpLiAgICANCg0KTGFzdGx5LCAwXnRoXi1vcmRlciB3YXMgdGhlIGxlYXN0IGNvbW1vbiAoKm4qID0gMywgNSUpLCBpdCBzdWdnZXN0cyB0aGF0IE1PPHN1Yj4yPC9zdWI+IGRvZXMgbm90IHNob3cgYSBzdGF0aXN0aWNhbGx5IHNpZ25pZmljYW50IGRlcGVuZGVuY2Ugb24gdGhlIE88c3ViPjI8L3N1Yj4uIEluIG90aGVyIHdvcmRzLCB0aGUgbWV0YWJvbGljIHJhdGUgZG9lcyBub3QgYWRqdXN0IGJhc2VkIG9uIG94eWdlbiBhdmFpbGFiaWxpdHksIGFuZCB0aGVyZSBpcyBubyBjbGVhciBjcml0aWNhbCBveHlnZW4gdGhyZXNob2xkIChQY3JpdCkgd2hlcmUgdGhlIHJlbGF0aW9uc2hpcCBjaGFuZ2VzLiBUaGlzIGlzIGluZGljYXRpdmUgb2YgYSAqKm94eXJlZ3VsYXRvcioqLiAgIA0KICANCmBgYHtyfQ0KYmVzdF9tb2QgPC0gYmVzdF9maXRfYmF5ZXNfcmVnICU+JSANCiAgZHBseXI6OmZpbHRlcihlbHBkX2xvb19yYW5rID09IDEpDQoNCnRvdGFsX2Zpc2ggPC0gbnJvdyhiZXN0X21vZCkNCg0KbW9kX3N1bW1hcnlfdGFibGUgPC0gYmVzdF9tb2QgJT4lIA0KICBkcGx5cjo6Z3JvdXBfYnkobW9kZWxfdHlwZSkgJT4lIA0KICBkcGx5cjo6cmVmcmFtZShuID0gbGVuZ3RoKGlkKSwNCiAgICAgICAgICAgICAgICAgcGVyY2VudCA9IHJvdW5kKChuL3RvdGFsX2Zpc2gpKjEwMCwyKSkgJT4lIA0KICBkcGx5cjo6bXV0YXRlKGJlc3RfbW9kZWxfbmFtZSA9IGNhc2Vfd2hlbigNCiAgICAgIG1vZGVsX3R5cGUgPT0gImxtXzAiIH4gIjB0aC1vcmRlciBwb2x5bm9taWFsIiwNCiAgICAgIG1vZGVsX3R5cGUgPT0gImxtXzEiIH4gIjFzdC1vcmRlciBwb2x5bm9taWFsIiwNCiAgICAgIG1vZGVsX3R5cGUgPT0gImxtXzIiIH4gIjJuZC1vcmRlciBwb2x5bm9taWFsIiwNCiAgICAgIG1vZGVsX3R5cGUgPT0gImxtXzMiIH4gIjNyZC1vcmRlciBwb2x5bm9taWFsIiwNCiAgICAgIFRSVUUgfiAiRVJST1IiDQogICAgKSkNCg0KDQp0YWJsZV9id20gPC0gbW9kX3N1bW1hcnlfdGFibGUgJT4lIA0KICBkcGx5cjo6c2VsZWN0KGJlc3RfbW9kZWxfbmFtZSwgZXZlcnl0aGluZygpLCAtbW9kZWxfdHlwZSkgJT4lIA0KICAgZ3QoKSAlPiUgDQogIGNvbHNfYWxpZ24oDQogICAgYWxpZ24gPSAiY2VudGVyIiwgDQogICAgY29sdW1ucyA9IGV2ZXJ5dGhpbmcoKQ0KICApDQoNCnRhYmxlX2J3bQ0KYGBgDQogIA0KU3VtbWFyeSBvZiBmaXNoIGJlc3QgbW9kZWwgd2l0aCBhIGxpbmVhciBmdW5jdGlvbi4gIA0KDQojIyMgVGFibGUgUzMgDQoNCioqVGFibGUgUzM6KiogVGVuIGZpc2ggdGhhdCB3ZXJlIGJlc3QgbW9kZWxsZWQgd2l0aCBhIGxpbmVhciBmdW5jdGlvbiwgc2hvd2luZyByXjJeLCBhbmQgZXN0aW1hdGUgKCkuIE9ubHkgdHdvIGZpc2ggaGFkIHBvc2l0aXZlIGVzdGltYXRlcyB3aXRoIGNyZWRpYmxlIGludGVydmFscyB0aGF0IGRpZCBub3Qgb3ZlcmxhcCB3aXRoIHplcm8sIHdoaWNoIGFyZSBoaWdobGlnaHRlZCBhcyBjb25mb3JtaW5nIGluIHRoZSB0YWJsZS4gVGh1cyBpbiB0b3RhbCwgMTMgb2YgNTggZmlzaCBzaG93ZWQgc3VmZmljaWVudCBldmlkZW5jZSB0byBiZSBjb25mb3JtaW5nLg0KDQpgYGB7cn0NCnRhYmxlX2xtXzEgPC0gYmVzdF9tb2QgJT4lIA0KICBkcGx5cjo6ZmlsdGVyKG1vZGVsX3R5cGUgPT0gImxtXzEiKSAlPiUgDQogIGRwbHlyOjptdXRhdGUocl9zcV9jaSA9IHBhc3RlMChyb3VuZChyMiwgMyksICIgKCIsIA0KICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgcm91bmQocjJfcTIuNSwgMyksICLigJMiLCANCiAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgIHJvdW5kKHIyX3E5Ny41LCAzKSwgIikiKSwNCiAgICAgICAgICAgICAgICBlc3RfY2kgPSBwYXN0ZTAocm91bmQoZXN0aW1hdGVfRE8sIDYpLCAiICgiLCANCiAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgcm91bmQoY29uZi5sb3dfRE8sIDYpLCAi4oCTIiwgDQogICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgIHJvdW5kKGNvbmYuaGlnaF9ETywgNiksICIpIiksDQogICAgICAgICAgICAgICAgY29uZm9ybWVyID0gaWZfZWxzZShjb25mLmxvd19ETyA+IDAsICJDb25mb3JtaW5nIiwgIk5vdCBjb25mb3JtaW5nIikpICU+JSANCiAgZHBseXI6OnNlbGVjdChpZCwgcl9zcV9jaSwgZXN0X2NpLCBjb25mb3JtZXIpICU+JSANCiAgZHBseXI6OmFycmFuZ2UoY29uZm9ybWVyKSAlPiUgDQogIGRwbHlyOjp1bmdyb3VwKCkNCg0KDQp0YWJsZV9sbV8xICU+JSANCiAgZ3QoKSAlPiUgDQogIGNvbHNfYWxpZ24oDQogICAgYWxpZ24gPSAiY2VudGVyIiwgDQogICAgY29sdW1ucyA9IGV2ZXJ5dGhpbmcoKQ0KICApICU+JSANCiAgY29sc19sYWJlbCgNCiAgICBpZCA9ICJGaXNoIElEIiwNCiAgICByX3NxX2NpID0gInIyIChDSSkiLA0KICAgIGVzdF9jaSA9ICJFc3RpbWF0ZSAoQ0kpIiwNCiAgICBjb25mb3JtZXIgPSAiRXZpZGVuY2Ugb2Ygb3h5Y29uZm9ybWluZyINCiAgKQ0KYGBgDQoNCiMjIyMgUGxvdGluZyBhbGwgbW9kZWxzDQoNCkRhdGEgc2V0IHdpdGggYWxsIHNsb3BlcyBhbmQgd2hpY2ggbW9kZWwgd2FzIGJlc3QNCg0KYGBge3J9DQpiZXN0X2ZpdCA8LSBsZWZ0X2pvaW4oc2xvcGVfdGlkeSwgYmVzdF9tb2QsIGJ5ID0gImlkIikNCmBgYA0KDQpOb3cgd2UgYXJlIHBsb3R0aW5nIGVhY2ggb2YgdGhlIHJlZ3Jlc3Npb25zLiBGaXJzdCBtYWtpbmcgYSBkaXJlY3RvcnkgdG8gc2F2ZSB0aGUgZmlndXJlcyAgIA0KDQpgYGB7ciwgZWNobz1GQUxTRX0NCmluY3JlbWVudGFsX3JlZ19iYXllc193ZCA8LSBmaWxlLnBhdGgob3V0cHV0X2ZpZ193ZCwgImluY3JlbWVudGFsX3JlZ3Jlc3Npb25zIikNCmlmICghZGlyLmV4aXN0cyhpbmNyZW1lbnRhbF9yZWdfYmF5ZXNfd2QpKSB7DQogIGRpci5jcmVhdGUoaW5jcmVtZW50YWxfcmVnX2JheWVzX3dkKQ0KfQ0KYGBgDQogIA0KUGxvdCBhbGwgcmVncmVzc2lvbiwgYW5kIGhpZ2hsaWdodGluZyB0aGUgbW9kZWwgdGhhdCBoYXMgdGhlIGJlc3QgZml0LCBiYXNlZCBvbiBBSUMgdmFsdWVzICAgDQoNCmBgYHtyLCBmaWcuaGVpZ2h0ID0gMywgZmlnLndpZHRoPSA0fQ0KIyBDcmVhdGUgYSBsaXN0IHRvIHN0b3JlIHRoZSBwbG90cw0KcGxvdHMgPC0gbGlzdCgpDQptb2RlbF9wcmVkc19saXN0IDwtIGxpc3QoKQ0KDQpmb3IgKGlkX2kgaW4gaWRzKSB7DQogIA0KICAjIEZpbHRlciBkYXRhIGZvciB0aGUgY3VycmVudCBJRA0KICBkZl9pIDwtIGJheWVzX3JlZ19tb2RzX3ByZWRpY3Rpb25zICU+JQ0KICAgIGRwbHlyOjpmaWx0ZXIoaWQgPT0gaWRfaSkgJT4lIA0KICAgIGRwbHlyOjptdXRhdGUobGluZV9zaXplID0gaWZfZWxzZShlbHBkX2xvb19yYW5rID09IDEsIDIsIDEpLA0KICAgICAgICAgICBhbHBoYV92YWx1ZSA9IGlmX2Vsc2UoZWxwZF9sb29fcmFuayA9PSAxLCAxLCAwLjQpKQ0KICANCiAgeF9taW4gPC0gZGZfaSAlPiUNCiAgICBkcGx5cjo6cmVmcmFtZShtaW4gPSBtaW4oRE8pLCBuYS5ybSA9IFRSVUUpICU+JSANCiAgICBkcGx5cjo6cHVsbChtaW4pDQogIA0KICB5X21heCA8LSBkZl9pICU+JQ0KICAgIGRwbHlyOjpyZWZyYW1lKG1heCA9IG1heChNTzJfZyksIG5hLnJtID0gVFJVRSkgJT4lIA0KICAgIGRwbHlyOjpwdWxsKG1heCkNCiAgDQogIGJlc3Rfd2VpZ2h0ZWRfbW9kZWxfaSA8LSBiZXN0X2ZpdF9iYXllc19yZWcgJT4lIA0KICAgIGRwbHlyOjpmaWx0ZXIoaWQgPT0gaWRfaSAmIGVscGRfbG9vX3JhbmsgPT0gMSkNCiAgDQogIHBvbHlfaV9uYW1lIDwtIGJlc3Rfd2VpZ2h0ZWRfbW9kZWxfaSAlPiUNCiAgICBkcGx5cjo6bXV0YXRlKG5hbWUgPSBjYXNlX3doZW4oDQogICAgICBtb2RlbF90eXBlID09ICJsbV8wIiB+ICIwdGgtb3JkZXIiLA0KICAgICAgbW9kZWxfdHlwZSA9PSAibG1fMSIgfiAiMXN0LW9yZGVyIiwNCiAgICAgIG1vZGVsX3R5cGUgPT0gImxtXzIiIH4gIjJuZC1vcmRlciIsDQogICAgICBtb2RlbF90eXBlID09ICJsbV8zIiB+ICIzcmQtb3JkZXIiLA0KICAgICAgVFJVRSB+ICJFUlJPUiINCiAgICApKSAlPiUgDQogICAgZHBseXI6OnB1bGwobmFtZSkNCiAgDQogIHJfaSA8LSBiZXN0X3dlaWdodGVkX21vZGVsX2kgJT4lIA0KICAgIGRwbHlyOjptdXRhdGUocl9zcV9jaSA9IHBhc3RlMChyb3VuZChyMiwgMyksICIgKCIsIA0KICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgcm91bmQocjJfcTIuNSwgMyksICLigJMiLCANCiAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgIHJvdW5kKHIyX3E5Ny41LCAzKSwgIikiKSkgJT4lIA0KICAgIGRwbHlyOjpwdWxsKHJfc3FfY2kpDQoNCiAgIyBDcmVhdGUgdGhlIHBsb3QNCiAgcGxvdF9pIDwtIGdncGxvdCgpICsNCiAgICBnZW9tX3JpYmJvbihkYXRhID0gZGZfaSwNCiAgICAgICAgICAgICAgICBhZXMoeCA9IERPLCB5ID0gcHJlZGljdGVkLCB5bWluID0gcHJlZF9sb3dlciwgeW1heCA9IHByZWRfdXBwZXIsIGZpbGwgPSBtb2RlbF90eXBlKSwNCiAgICAgICAgICAgICAgICBhbHBoYSA9IDAuMSkgKw0KICAgIGdlb21fbGluZShkYXRhID0gZGZfaSwgDQogICAgICAgICAgICAgIGFlcyh4ID0gRE8sIHkgPSBwcmVkaWN0ZWQsIGNvbG91ciA9IG1vZGVsX3R5cGUsIHNpemUgPSBsaW5lX3NpemUsIGFscGhhID0gYWxwaGFfdmFsdWUpKSArDQogICAgZ2VvbV9wb2ludChkYXRhID0gZGZfaSAlPiUgZHBseXI6OmZpbHRlcihlbHBkX2xvb19yYW5rID09IDEpLCBhZXMoeCA9IERPLCB5ID0gTU8yX2cpLCBhbHBoYSA9IDAuNiwgY29sb3VyID0gImJsYWNrIiwgc2l6ZSA9IDIpICsNCiAgICBzY2FsZV9jb2xvdXJfbWFudWFsKHZhbHVlcyA9IGMoInJlZCIsICJibHVlIiwgImdyZWVuIiwgInB1cnBsZSIpLCANCiAgICAgICAgICAgICAgICAgICAgICAgIGxhYmVscyA9IGMoIjB0aCBPcmRlciIsICIxc3QgT3JkZXIiLCAiMm5kIE9yZGVyIiwgIjNyZCBPcmRlciIpKSArDQogICAgc2NhbGVfc2l6ZV9pZGVudGl0eSgpICsgICMgVXNlIHRoZSBzaXplIHZhbHVlcyBkaXJlY3RseQ0KICAgIHNjYWxlX2FscGhhX2lkZW50aXR5KGd1aWRlID0gIm5vbmUiKSArICAjIFJlbW92ZSB0aGUgYWxwaGEgbGVnZW5kIA0KICAgIGFubm90YXRlKCJ0ZXh0IiwgeCA9IHhfbWluLCANCiAgICAgICAgICAgICB5ID0geV9tYXgsIA0KICAgICAgICAgICAgIGxhYmVsID0gcGFzdGUwKCJCZXN0IGZpdDogIixwb2x5X2lfbmFtZSwgIlxuIiwgInIyID0gIiwgcl9pKSwgDQogICAgICAgICAgICAgaGp1c3QgPSAwLCB2anVzdCA9IDEsIHNpemUgPSA0KSArDQogICAgbGFicygNCiAgICAgIHRpdGxlID0gcGFzdGUoaWRfaSksDQogICAgICB4ID0gIkRpc3NvbHZlZCBveHlnZW4gcGVyY2VudGFnZSAoRE8pIiwNCiAgICAgIHkgPSAiTU8yIChvMiBtZy9nL2gpIiwNCiAgICAgIGNvbG91ciA9ICJNb2RlbCIpICsNCiAgICB0aGVtZV9jbGFzc2ljKCkgKw0KICAgIHRoZW1lKGxlZ2VuZC5wb3NpdGlvbiA9ICJub25lIikNCiAgDQogICMgU3RvcmUgdGhlIHBsb3QNCiAgcGxvdHNbW2lkX2ldXSA8LSBwbG90X2kNCiAgDQogIHByaW50KHBsb3RfaSkNCn0NCg0KDQpwZGYoZmlsZSA9IHBhc3RlMChpbmNyZW1lbnRhbF9yZWdfYmF5ZXNfd2QsICIuL2NvbWJpbmVkX3JlZ19wbG90cy5wZGYiKSwgd2lkdGggPSA4LCBoZWlnaHQgPSA2KQ0KDQpmb3IgKGlkX2kgaW4gaWRzKSB7DQogIHByaW50KHBsb3RzW1tpZF9pXV0pICAjIFByaW50IGVhY2ggcGxvdCB0byB0aGUgUERGDQp9DQoNCmRldi5vZmYoKSAgIyBDbG9zZSB0aGUgUERGIGRldmljZQ0KYGBgDQoNCkdldHRpbmcgYSBwbG90IGZvciBlYWNoIGJlc3QgZml0IHJlZ3Jlc3Npb24sIGFuZCBvdmVybGF5aW5nIGEgZ2xvYmFsIG1vZGVsIGZvciB0aGF0IG1vZGVsIHR5cGUuDQoNCmBgYHtyLCBlY2hvPUZBTFNFfQ0Kb3V0cHV0X21vZHNfYmF5ZXNfZ2xvYmFsX3dkIDwtIHBhc3RlMChvdXRwdXRfbW9kc193ZCwgIi4vYmF5ZXMtcmVncy1nbG9iYWwiKQ0KaWYgKCFkaXIuZXhpc3RzKG91dHB1dF9tb2RzX2JheWVzX2dsb2JhbF93ZCkpIHtkaXIuY3JlYXRlKG91dHB1dF9tb2RzX2JheWVzX2dsb2JhbF93ZCl9DQpgYGANCg0KSGVyZSB3ZSBhcmUgZ3JvdXBpbmcgZmlzaCBieSBiZXN0IGZpdHRpbmcgbW9kZWwgYW5kIGdldHRpbmcgYW4gYXZlcmFnZSB0cmVuZC4gSSBoYXZlIGhhc2hlZCB0aGUgY29kZSBzbyB5b3UgZG9uJ3QgbmVlZCB0byByZS1ydW4gdGhlIG1vZGVscw0KDQrij63vuI8gKipTa2lwIHRoaXMgc3RlcCoqIGlmIHlvdSBoYXZlIGRvd25sb2FkZWQgdGhlIGdsb2JhbCBtb2RlbHMgKionLnJkcycqKiBpbiB0aGUgJ2JheWVzLXJlZ3MtZ2xvYmFsJyBmb2xkZXINCg0KYGBge3J9DQojIHNldHdkKG91dHB1dF9tb2RzX2JheWVzX2dsb2JhbF93ZCkNCiMgDQojIHNlZWRfbnVtYmVyID0gMTQzMDE5DQojIA0KIyAjIFNldCB1cCBwYXJhbGxlbCBiYWNrZW5kDQojIHBsYW4obXVsdGlzZXNzaW9uLCB3b3JrZXJzID0gNCkgICMgQWRqdXN0IHdvcmtlcnMgYmFzZWQgb24gc3lzdGVtIHJlc291cmNlcw0KIyANCiMgIyBEZWZpbmUgbW9kZWwgZm9ybXVsYXMgYW5kIGRhdGENCiMgbW9kZWxfZm9ybXVsYXMgPC0gbGlzdCgNCiMgICBsbV8wID0gYmYoTU8yX2cgfiAxLCBmYW1pbHkgPSBnYXVzc2lhbigpKSwNCiMgICBsbV8xID0gYmYoTU8yX2cgfiBETywgZmFtaWx5ID0gZ2F1c3NpYW4oKSksDQojICAgbG1fMiA9IGJmKE1PMl9nIH4gcG9seShETywgMiksIGZhbWlseSA9IGdhdXNzaWFuKCkpLA0KIyAgIGxtXzMgPSBiZihNTzJfZyB+IHBvbHkoRE8sIDMpLCBmYW1pbHkgPSBnYXVzc2lhbigpKQ0KIyApDQojIA0KIyBtb2RlbF9kYXRhIDwtIGxpc3QoDQojICAgbG1fMCA9IGJlc3RfZml0ICU+JSBmaWx0ZXIobW9kZWxfdHlwZSA9PSAibG1fMCIpLA0KIyAgIGxtXzEgPSBiZXN0X2ZpdCAlPiUgZmlsdGVyKG1vZGVsX3R5cGUgPT0gImxtXzEiKSwNCiMgICBsbV8yID0gYmVzdF9maXQgJT4lIGZpbHRlcihtb2RlbF90eXBlID09ICJsbV8yIiksDQojICAgbG1fMyA9IGJlc3RfZml0ICU+JSBmaWx0ZXIobW9kZWxfdHlwZSA9PSAibG1fMyIpDQojICkNCiMgDQojICMgUnVuIG1vZGVscyBpbiBwYXJhbGxlbCB3aXRoIGZ1dHVyZV9tYXAoKQ0KIyBtb2RlbHMgPC0gZnV0dXJlX21hcChuYW1lcyhtb2RlbF9mb3JtdWxhcyksIH57DQojICAgYnJtKA0KIyAgICAgZm9ybXVsYSA9IG1vZGVsX2Zvcm11bGFzW1sueF1dLA0KIyAgICAgZGF0YSA9IG1vZGVsX2RhdGFbWy54XV0sDQojICAgICBjb3JlcyA9IDQsDQojICAgICBzZWVkID0gc2VlZF9udW1iZXIsDQojICAgICBzYXZlX3BhcnMgPSBzYXZlX3BhcnMoYWxsID0gVFJVRSksDQojICAgICBzYW1wbGVfcHJpb3IgPSBGQUxTRSwNCiMgICAgIHNpbGVudCA9IFRSVUUsDQojICAgICBmaWxlID0gcGFzdGUwKC54LCAiX2dsb2JhbCIpDQojICAgKQ0KIyB9KQ0KIyANCiMgIyBBc3NpZ24gbW9kZWwgbmFtZXMgdG8gcmVzdWx0cw0KIyBuYW1lcyhtb2RlbHMpIDwtIG5hbWVzKG1vZGVsX2Zvcm11bGFzKQ0KIyANCiMgIyBTdG9wIHBhcmFsbGVsIHBsYW4gYWZ0ZXIgZXhlY3V0aW9uDQojIHBsYW4oc2VxdWVudGlhbCkNCmBgYA0KDQpMb2FkaW5nIHRoZXNlIG1vZGVscw0KDQpgYGB7cn0NCmJheWVzX3JlZ19tb2RzX2dsb2JhbCA8LSBsb2FkX3Jkcyhtb2RlbF9kdyA9IG91dHB1dF9tb2RzX2JheWVzX2dsb2JhbF93ZCkNCmBgYA0KDQpQdWxsaW5nIHByZWRpY3Rpb25zDQoNCmBgYHtyfQ0KZ2xvYmFsX21vZGVsc19wcmVkcyA8LSBsaXN0KCkNCg0KZm9yIChtb2RlX25hbWUgaW4gbmFtZXMoYmF5ZXNfcmVnX21vZHNfZ2xvYmFsKSkgew0KICANCiAgbW9kX2kgPC0gYmF5ZXNfcmVnX21vZHNfZ2xvYmFsW1ttb2RlX25hbWVdXQ0KICBtb2RlX3R5cGUgPSBzdHJfcmVtb3ZlKG1vZGVfbmFtZSwgIl9nbG9iYWwiKQ0KICANCiAgIG1vZGVsX3ByZWRpY3Rpb25zX2kgPC0gYXMuZGF0YS5mcmFtZShmaXR0ZWQobW9kX2ksIHN1bW1hcnkgPSBUUlVFKSkgJT4lIA0KICAgICAgZHBseXI6Om11dGF0ZShtb2RlbCA9IG1vZGVfbmFtZSwNCiAgICAgICAgICAgICAgICAgICAgbW9kZWxfdHlwZSA9IG1vZGVfdHlwZSkgJT4lIA0KICAgICAgZHBseXI6OnJlbmFtZShwcmVkX2xvd2VyID0gUTIuNSwgcHJlZF91cHBlciA9IFE5Ny41LCBwcmVkaWN0ZWQgPSBFc3RpbWF0ZSwgcHJlZF9lcnJvciA9IEVzdC5FcnJvcikgJT4lIA0KICAgICAgZHBseXI6OnNlbGVjdChtb2RlbCwgZXZlcnl0aGluZygpKQ0KICAgDQogICBvcmlnaW5hbF9kYXRhX2kgPC0gYmVzdF9maXQgJT4lIGZpbHRlcihtb2RlbF90eXBlID09IG1vZGVfdHlwZSkgJT4lIA0KICAgICBkcGx5cjo6c2VsZWN0KC1tb2RlbF90eXBlKQ0KICAgIA0KICAgIG1vZGVsX3ByZWRpY3Rpb25zX29yaWdpbmFsX2kgPC0gY2JpbmQobW9kZWxfcHJlZGljdGlvbnNfaSwgb3JpZ2luYWxfZGF0YV9pKQ0KICAgIA0KICAgDQogICBnbG9iYWxfbW9kZWxzX3ByZWRzW1ttb2RlX25hbWVdXSA8LSBtb2RlbF9wcmVkaWN0aW9uc19vcmlnaW5hbF9pDQp9DQoNCmdsb2JhbF9tb2RlbHNfcHJlZF9kZiA8LSBiaW5kX3Jvd3MoZ2xvYmFsX21vZGVsc19wcmVkcykNCmBgYA0KDQojIyMjIEZpZ3VyZSBTMTANCg0KQmVzdCBmaXQgcmVncmVzc2lvbnMgd2l0aCBnbG9iYWwgbW9kZWxzIGJhc2VkIG9uIHRoYXQgcG9seW5vbWlhbCBvcmRlci4NCg0KRmlndXJlIFMxMCBpcyBhbHNvIGNhbGxlZCAqKkZpZ3VyZSAyKiogaW4gdGhlIG1hbnVzY3JpcHQuIA0KDQoNCmBgYHtyfQ0KZ2xvYmFsX21vZGVsc19wcmVkX2RmIDwtIGdsb2JhbF9tb2RlbHNfcHJlZF9kZiAlPiUNCiAgZHBseXI6Om11dGF0ZShtb2RlbF9uYW1lID0gY2FzZV93aGVuKA0KICAgICAgbW9kZWxfdHlwZSA9PSAibG1fMCIgfiAiMHRoLW9yZGVyIHBvbHlub21pYWwiLA0KICAgICAgbW9kZWxfdHlwZSA9PSAibG1fMSIgfiAiMXN0LW9yZGVyIHBvbHlub21pYWwiLA0KICAgICAgbW9kZWxfdHlwZSA9PSAibG1fMiIgfiAiMm5kLW9yZGVyIHBvbHlub21pYWwiLA0KICAgICAgbW9kZWxfdHlwZSA9PSAibG1fMyIgfiAiM3JkLW9yZGVyIHBvbHlub21pYWwiLA0KICAgICAgVFJVRSB+ICJFUlJPUiINCiAgICApKQ0KDQphbm5vdGF0aW9uX2RhdGEgPC0gbW9kX3N1bW1hcnlfdGFibGUgJT4lDQogIGRwbHlyOjpzZWxlY3QobW9kZWxfdHlwZSwgYmVzdF9tb2RlbF9uYW1lLCBuKQ0KDQpiYXllc19yZWdfbW9kc19wcmVkaWN0aW9uc19iZXN0IDwtIGJheWVzX3JlZ19tb2RzX3ByZWRpY3Rpb25zICU+JSANCiAgZHBseXI6OmZpbHRlcihlbHBkX2xvb19yYW5rID09IDEpIA0KDQpmaWdfMSA8LSBnZ3Bsb3QoKSArDQogIGdlb21fbGluZShkYXRhID0gYmF5ZXNfcmVnX21vZHNfcHJlZGljdGlvbnNfYmVzdCwNCiAgICAgICAgICAgIGFlcyh4ID0gRE8sIHkgPSBwcmVkaWN0ZWQsIGNvbG9yID0gaWQpLCBzaXplID0gMSkgKw0KICAjIGdlb21fc21vb3RoKGRhdGEgPSBiZXN0X2ZpdCwNCiAgIyAgICAgICAgICAgYWVzKHggPSBETywgeSA9IE1PMl9nLCBjb2xvciA9IGlkKSwgc2l6ZSA9IDEsIGFscGhhID0gMSwgc2UgPSBGQUxTRSkgKw0KICBnZW9tX3JpYmJvbihkYXRhID0gZ2xvYmFsX21vZGVsc19wcmVkX2RmLA0KICAgICAgICAgICAgICBhZXMoeCA9IERPLCB5bWluID0gcHJlZF9sb3dlcg0KLCB5bWF4ID0gcHJlZF91cHBlciwgZ3JvdXAgPSBtb2RlbCksDQogICAgICAgICAgICAgIGZpbGwgPSAiI0ZDNkM4NSIsIGFscGhhID0gMC4yKSArICAjIFNoYWRlZCBjb25maWRlbmNlIGludGVydmFscw0KICBnZW9tX2xpbmUoZGF0YSA9IGdsb2JhbF9tb2RlbHNfcHJlZF9kZiwNCiAgICAgICAgICAgIGFlcyh4ID0gRE8sIHkgPSBwcmVkaWN0ZWQpLCBzaXplID0gMS41LCBjb2xvciA9ICIjRkYwMDdGIikgKw0KICBmYWNldF93cmFwKH5tb2RlbF90eXBlKSArDQogIHNjYWxlX2NvbG9yX2dyZXkoc3RhcnQgPSAwLjEsIGVuZCA9IDAuOSkgKw0KICBsYWJzKA0KICAgICAgdGl0bGUgPSBwYXN0ZSgiQmVzdCBmaXQgcmVncmVzc2lvbnMgZ3JvdXBlZCBieSBwb2x5bm9taWFsIG9yZGVyIiksDQogICAgICB4ID0gIkRpc3NvbHZlZCBveHlnZW4gcGVyY2VudGFnZSAoRE8pIiwNCiAgICAgIHkgPSAiTU8yIChPMiBtZy9nL2gpIikgKw0KICB0aGVtZV9jbGFzc2ljKCkgKw0KICB0aGVtZShsZWdlbmQucG9zaXRpb24gPSAibm9uZSIpICsNCiAgZ2VvbV90ZXh0KGRhdGEgPSBhbm5vdGF0aW9uX2RhdGEsDQogICAgICAgICAgICBhZXMoeCA9IC1JbmYsIHkgPSBJbmYsIGxhYmVsID0gcGFzdGUwKCJpdGFsaWMobikgPT0gIiwgbikpLA0KICAgICAgICAgICAgaGp1c3QgPSAtMC4xLCB2anVzdCA9IDEuMiwgaW5oZXJpdC5hZXMgPSBGQUxTRSwgcGFyc2UgPSBUUlVFKQ0KZmlnXzENCmBgYA0KDQoqKkZpZ3VyZSAyOioqIEJlc3QgZml0IG1vZGVscyBmcm9tIGluY3JlbWVudGFsIHJlZ3Jlc3Npb24gYW5hbHlzZXMgKDBedGheIHRvIDMgXnJkXiBwb2x5bm9taWFsIHJlZ3Jlc3Npb25zKSBncm91cGVkIGJ5IHBvbHlub21pYWwgb3JkZXIsIHRoZSBwaW5rIHJlZ3Jlc3Npb24gcmVwcmVzZW50cyB0aGUgYXZlcmFnZSBmb3IgYWxsIGZpc2ggZml0IHdpdGggdGhhdCBwb2x5bm9taWFsIG9yZGVyLg0KDQojIyBQY3JpdCBtb2RlbA0KDQpGb3IgdGhvc2UgZmlzaCB0aGF0IHdlcmUgYmVzdCBtb2RlbGxlZCB3aXRoIGEgMl5uZF4gb3IgM15yZF4tb3JkZXIgcG9seW5vbWlhbCAoKm4qID0gMzcpIHdlIHdpbGwgY2hlY2sgdG8gc2VlIGlmIGEgUGNyaXQgaXMgcHJlc2VudC4gV2UgYXJlIGZpbHRlcmluZyB0aGUgZGF0YSBmb3Igb25seSB0aG9zZSBmaXNoLiAgIA0KDQpgYGB7cn0NCmhpZ2hlcl9vcmRlciA8LSBjKCJsbV8yIiwgImxtXzMiKQ0KDQpjaGVja19wY3JpdF9pZHMgPC0gYmVzdF9tb2QgJT4lIA0KICBkcGx5cjo6ZmlsdGVyKG1vZGVsX3R5cGUgJWluJSBoaWdoZXJfb3JkZXIpICU+JSANCiAgZHBseXI6OmRpc3RpbmN0KGlkKSAlPiUgDQogIHB1bGwoaWQpDQoNCmNoZWNrX3Bjcml0X2RmIDwtIHNsb3BlX3RpZHkgJT4lIA0KICBkcGx5cjo6ZmlsdGVyKGlkICVpbiUgY2hlY2tfcGNyaXRfaWRzKQ0KYGBgDQogIA0KV2Ugd2lsbCBjYWxjdWxhdGUgUGNyaXQgdXNpbmcgQ2hhYm90IG1ldGhvZCBhbmQgZnVuY3Rpb24gYGNhbGNPMmNyaXRgLg0KDQpUaGlzIGZ1bmN0aW9uIHVzZXMgdGhlIGZpZnRoIHBlcmNlbnRpbGUgb2YgdGhlIE1PPHN1Yj4yPC9zdWI+IHZhbHVlcyBvYnNlcnZlZCBhdCBkaXNzb2x2ZWQgb3h5Z2VuIGxldmVscyDiiaUgODAlIGFpciBzYXR1cmF0aW9uIGFzIHRoZSBjcml0ZXJpb24gdG8gYXNzZXNzIGxvdyAgTU88c3ViPjI8L3N1Yj4gdmFsdWVzLiBUaGUgYWxnb3JpdGhtIHRoZW4gaWRlbnRpZmllcyBhbGwgdGhlIE1PPHN1Yj4yPC9zdWI+IG1lYXN1cmVtZW50cyBncmVhdGVyIHRoYW4gdGhpcyBtaW5pbWFsbHkgYWNjZXB0YWJsZSBNTzxzdWI+Mjwvc3ViPiB2YWx1ZS4gV2l0aGluIHRoaXMgc3ViLXNldCwgaXQgaWRlbnRpZmllcyB0aGUgzIcgTU8yIG1lYXN1cmVtZW50IG1hZGUgYXQgdGhlIGxvd2VzdCBETyBhbmQgdGhlcmVhZnRlciBjb25zaWRlcnMgdGhpcyBETyBhcyBjYW5kaWRhdGUgZm9yIGJyZWFrcG9pbnQgKG5hbWVkIHBpdm90RE8gaW4gdGhlIHNjcmlwdCkuIEEgcmVncmVzc2lvbiBpcyB0aGVuIGNhbGN1bGF0ZWQgdXNpbmcgb2JzZXJ2YXRpb25zIGF0IERPIGxldmVscyA8IHBpdm90RE8sIGFuZCBhIGZpcnN0IGVzdGltYXRlIG9mIE8yY3JpdCBpcyBjYWxjdWxhdGVkIGFzIHRoZSBpbnRlcnNlY3Rpb24gb2YgdGhpcyByZWdyZXNzaW9uIGxpbmUgd2l0aCB0aGUgaG9yaXpvbnRhbCBsaW5lIHJlcHJlc2VudGluZyBTTVIuIFRoZSBzY3JpcHQgdGhlbiBnb2VzIHRocm91Z2ggdmFsaWRhdGlvbiBzdGVwcyB0byBlbnN1cmUgdGhhdCB0aGUgc2xvcGUgb2YgdGhlIHJlZ3Jlc3Npb24gaXMgbm90IHNvIGxvdyB0aGF0IHRoZSBsaW5lLCBwcm9qZWN0ZWQgdG8gbm9ybW94aWMgRE8gbGV2ZWxzLCBwYXNzZXMgYmVsb3cgYW55IE1PMiB2YWx1ZXMgb2JzZXJ2ZWQgaW4gbm9ybW94aWEuIEl0IGFsc28gZW5zdXJlcyB0aGF0IHRoZSBpbnRlcmNlcHQgaXMgbm90IGdyZWF0ZXIgdGhhbiB6ZXJvLiBDb3JyZWN0aXZlIG1lYXN1cmVzIGFyZSB0YWtlbiBpZiBzdWNoIHByb2JsZW1zIGFyZSBlbmNvdW50ZXJlZC4gIA0KDQpsb3dlc3RNTzIgZGVmYXVsdCBpcyB0aGUgYHF1YW50aWxlKE1PMltETyA+PSA4MF0sIHA9MC4wNSlgLiBJdCBpcyB1c2VkIHRvIHNlZ21lbnQgdGhlIGRhdGEgYW5kIGxvY2F0ZSB0aGUgcGl2b3RETy4gICANCg0KYGBge3J9DQppZHMgPC0gY2hlY2tfcGNyaXRfZGYgJT4lIA0KICBkcGx5cjo6ZGlzdGluY3QoaWQpICU+JSANCiAgZHBseXI6OnB1bGwoKQ0KDQpwY3JpdF9tb2RlbF9kZl9saXN0IDwtIGxpc3QoKQ0KcGNyaXRfbW9kZWxzIDwtICBsaXN0KCkNCg0KZm9yIChpZF9pIGluIGlkcykgew0KDQpkZl9pIDwtIGNoZWNrX3Bjcml0X2RmICU+JSANCiAgZHBseXI6OmZpbHRlcihpZCA9PSBpZF9pKQ0KDQpvMmNyaXQgPC0gY2FsY08yY3JpdChEYXRhID0gZGZfaSwgU01SID0gZGZfaSRTTVJbMV0sIGxvd2VzdE1PMj1OQSwgZ2FwTGltaXQgPSA0LA0KICAgICAgICAgICAgICAgICAgICAgbWF4Lm5iLk1PMi5mb3IucmVnID0gNykNCg0KdmF1bGUgPC0gbzJjcml0JG8yY3JpdA0KbG93ZXN0TU8yID0gcXVhbnRpbGUoZGZfaSRNTzJbZGZfaSRETyA+PSA4MF0sIHA9MC4wNSkNClNNUiA8LSBvMmNyaXQkU01SDQpuYl9tbzJfY29uZm9ybWluZyA8LSBvMmNyaXQkTmJfTU8yX2NvbmZvcm1pbmcNCnIyIDwtIG8yY3JpdCRyMg0KbWV0aG9kIDwtIG8yY3JpdCRNZXRob2QNCnAgPC0gbzJjcml0JFBbMV0NCg0KcGNyaXRfbW9kZWxfZGYgPC0gdGliYmxlKA0KICAgICAgaWQgPSBpZF9pLA0KICAgICAgcGNyaXRfdmF1bGUgPSB2YXVsZSwNCiAgICAgIHBjcml0X3NtciA9IFNNUiwNCiAgICAgIHBjcml0X2xvd2VzdE1PMiA9IGxvd2VzdE1PMiwNCiAgICAgIHBjcml0X25iX21vMl9jb25mb3JtaW5nID0gbmJfbW8yX2NvbmZvcm1pbmcsDQogICAgICBwY3JpdF9yMiA9IHIyLA0KICAgICAgcGNyaXRfbWV0aG9kID0gbWV0aG9kLA0KICAgICAgcGNyaXRfcCA9IHANCiAgICApDQoNCnBjcml0X21vZGVsX2RmX2xpc3RbW2lkX2ldXSA8LSBwY3JpdF9tb2RlbF9kZg0KDQpwY3JpdF9tb2RlbHNbW2lkX2ldXSA8LSBvMmNyaXQNCg0KfQ0KDQpwY3JpdF9tb2RlbF9kZiA8LSBiaW5kX3Jvd3MocGNyaXRfbW9kZWxfZGZfbGlzdCkNCmBgYA0KDQojIyMgUGxvdGluZyBQY3JpdA0KDQpIZXJlJ3MgdGhlIHBsb3RzIGZvciB0aGUgUGNyaXQgZXN0aW1hdGVzLiBUaGlzIGluY2x1ZGVzIGFsbCBmaXNoIHdpdGggYmVzdCBmaXQgcmVncmVzc2lvbiB0aGF0IGFyZSAyIG9yIDMgb3JkZXIgKCpuKiA9IDM3KS4gICANCg0KYGBge3IsIGVjaG89RkFMU0V9DQojIENyZWF0ZSBvdXRwdXQgZGlyZWN0b3J5IGlmIG5lZWRlZA0Kb3V0cHV0X2ZpZ19wY3JpdF9jaGFib3Rfd2QgPC0gZmlsZS5wYXRoKG91dHB1dF9maWdfd2QsICJtb2RlbF9jaGFib3QiKQ0KaWYgKCFkaXIuZXhpc3RzKG91dHB1dF9maWdfcGNyaXRfY2hhYm90X3dkKSkgew0KICBkaXIuY3JlYXRlKG91dHB1dF9maWdfcGNyaXRfY2hhYm90X3dkKQ0KfQ0KDQppZHMgPC0gY2hlY2tfcGNyaXRfZGYgJT4lIA0KICBkcGx5cjo6ZGlzdGluY3QoaWQpICU+JSANCiAgZHBseXI6OnB1bGwoKQ0KDQpwY3JpdF9jaGFib3RfbGlzdCA8LSBsaXN0KCkNCg0KIyBPcGVuIGEgc2luZ2xlIFBERiBkZXZpY2UNCnBkZihmaWxlID0gZmlsZS5wYXRoKG91dHB1dF9maWdfcGNyaXRfY2hhYm90X3dkLCAiY29tYmluZWRfY2hhYm90X3Bsb3RzLnBkZiIpLCANCiAgICB3aWR0aCA9IDgsIGhlaWdodCA9IDYpDQoNCmZvciAoaWRfaSBpbiBpZHMpIHsNCiAgDQogIHIyIDwtIHBjcml0X21vZGVsX2RmICU+JSANCiAgICBkcGx5cjo6ZmlsdGVyKGlkID09IGlkX2kpICU+JSANCiAgICBkcGx5cjo6bXV0YXRlKHBjcml0X3IyID0gcm91bmQocGNyaXRfcjIsIDMpKSAlPiUgDQogICAgZHBseXI6OnB1bGwocGNyaXRfcjIpDQogIA0KICBjb25mb3JtaW5nIDwtIHBjcml0X21vZGVsX2RmICU+JSANCiAgICBkcGx5cjo6ZmlsdGVyKGlkID09IGlkX2kpICU+JSANCiAgICBkcGx5cjo6bXV0YXRlKHBjcml0X25iX21vMl9jb25mb3JtaW5nID0gcm91bmQocGNyaXRfbmJfbW8yX2NvbmZvcm1pbmcsIDMpKSAlPiUgDQogICAgZHBseXI6OnB1bGwocGNyaXRfbmJfbW8yX2NvbmZvcm1pbmcpDQogIA0KICBQIDwtIHBjcml0X21vZGVsX2RmICU+JSANCiAgICBkcGx5cjo6ZmlsdGVyKGlkID09IGlkX2kpICU+JSANCiAgICBkcGx5cjo6bXV0YXRlKHBjcml0X3AgPSByb3VuZChwY3JpdF9wLCAzKSkgJT4lIA0KICAgIGRwbHlyOjpwdWxsKHBjcml0X3ApDQogIA0KICBTTVIgPC0gcGNyaXRfbW9kZWxfZGYgJT4lIA0KICAgIGRwbHlyOjpmaWx0ZXIoaWQgPT0gaWRfaSkgJT4lIA0KICAgIGRwbHlyOjptdXRhdGUocGNyaXRfc21yID0gcm91bmQocGNyaXRfc21yLCAzKSkgJT4lIA0KICAgIGRwbHlyOjpwdWxsKHBjcml0X3NtcikNCiAgDQogIGxvd2VzdE1PMiA8LSBwY3JpdF9tb2RlbF9kZiAlPiUgDQogICAgZHBseXI6OmZpbHRlcihpZCA9PSBpZF9pKSAlPiUgDQogICAgZHBseXI6Om11dGF0ZShwY3JpdF9sb3dlc3RNTzIgPSByb3VuZChwY3JpdF9sb3dlc3RNTzIsIDMpKSAlPiUgDQogICAgZHBseXI6OnB1bGwocGNyaXRfbG93ZXN0TU8yKQ0KICANCiAgIyBHZW5lcmF0ZSBhbmQgcmVuZGVyIHRoZSBwbG90DQogIHBsb3RPMmNyaXQobzJjcml0b2JqID0gcGNyaXRfbW9kZWxzW1tpZF9pXV0pDQogIA0KICAjIEFkZCBhIHRpdGxlDQogIG10ZXh0KA0KICAgIHRleHQgPSBwYXN0ZTAoaWRfaSksDQogICAgc2lkZSA9IDMsIGxpbmUgPSAyLCBhZGogPSAwLA0KICAgIGNvbCA9ICJibHVlIiwgZm9udCA9IDIsIGNleCA9IDEuMg0KICApDQogIA0KICBtdGV4dCgNCiAgICB0ZXh0ID0gcGFzdGUwKCJSMiA9ICIsIHIyLCAiOyBwID0gIiwgUCwgIjsgQ1AgPCBTTVIgPSAiLCBjb25mb3JtaW5nLCAiOyBTTVIgPSAiLCBTTVIsICI7IGxvd2VzdE1PMiA9ICIsbG93ZXN0TU8yKSwNCiAgICBzaWRlID0gMywgbGluZSA9IDEsIGFkaiA9IDAsDQogICAgY29sID0gImJsdWUiLCBmb250ID0gMSwgY2V4ID0gMC44DQogICkNCn0NCg0KIyBDbG9zZSB0aGUgUERGIGRldmljZSAqYWZ0ZXIqIHRoZSBsb29wDQpkZXYub2ZmKCkNCmBgYA0KICANClByaW50aW5nIGluIGh0bG0gZG9jdW1lbnQgICANCg0KYGBge3IsIGZpZy5oZWlnaHQgPSAzLCBmaWcud2lkdGg9IDR9DQppZHMgPC0gY2hlY2tfcGNyaXRfZGYgJT4lIA0KICBkcGx5cjo6ZGlzdGluY3QoaWQpICU+JSANCiAgZHBseXI6OnB1bGwoKQ0KDQpmb3IgKGlkX2kgaW4gaWRzKSB7DQogIA0KICBjb21tZW50IDwtIGNoZWNrX3Bjcml0X2RmICU+JSANCiAgICBkcGx5cjo6ZmlsdGVyKGlkID09IGlkX2kpICU+JSANCiAgICBkcGx5cjo6c2xpY2UoMSkgJT4lIA0KICAgIGRwbHlyOjptdXRhdGUoY29tbWVudCA9IGlmX2Vsc2UoaXMubmEoY29tbWVudHMpLCAiIiwgcGFzdGUwKCIjIiwgY29tbWVudHMpKSkgJT4lIA0KICAgIHB1bGwoY29tbWVudCkNCiAgDQogIHIyIDwtIHBjcml0X21vZGVsX2RmICU+JSANCiAgICBkcGx5cjo6ZmlsdGVyKGlkID09IGlkX2kpICU+JSANCiAgICBkcGx5cjo6bXV0YXRlKHBjcml0X3IyID0gcm91bmQocGNyaXRfcjIsIDMpKSAlPiUgDQogICAgZHBseXI6OnB1bGwocGNyaXRfcjIpDQogIA0KICBjb25mb3JtaW5nIDwtIHBjcml0X21vZGVsX2RmICU+JSANCiAgICBkcGx5cjo6ZmlsdGVyKGlkID09IGlkX2kpICU+JSANCiAgICBkcGx5cjo6bXV0YXRlKHBjcml0X25iX21vMl9jb25mb3JtaW5nID0gcm91bmQocGNyaXRfbmJfbW8yX2NvbmZvcm1pbmcsIDMpKSAlPiUgDQogICAgZHBseXI6OnB1bGwocGNyaXRfbmJfbW8yX2NvbmZvcm1pbmcpDQogIA0KICBQIDwtIHBjcml0X21vZGVsX2RmICU+JSANCiAgICBkcGx5cjo6ZmlsdGVyKGlkID09IGlkX2kpICU+JSANCiAgICBkcGx5cjo6bXV0YXRlKHBjcml0X3AgPSByb3VuZChwY3JpdF9wLCAzKSkgJT4lIA0KICAgIGRwbHlyOjpwdWxsKHBjcml0X3ApDQogIA0KICBTTVIgPC0gcGNyaXRfbW9kZWxfZGYgJT4lIA0KICAgIGRwbHlyOjpmaWx0ZXIoaWQgPT0gaWRfaSkgJT4lIA0KICAgIGRwbHlyOjptdXRhdGUocGNyaXRfc21yID0gcm91bmQocGNyaXRfc21yLCAzKSkgJT4lIA0KICAgIGRwbHlyOjpwdWxsKHBjcml0X3NtcikNCiAgDQogIGxvd2VzdE1PMiA8LSBwY3JpdF9tb2RlbF9kZiAlPiUgDQogICAgZHBseXI6OmZpbHRlcihpZCA9PSBpZF9pKSAlPiUgDQogICAgZHBseXI6Om11dGF0ZShwY3JpdF9sb3dlc3RNTzIgPSByb3VuZChwY3JpdF9sb3dlc3RNTzIsIDMpKSAlPiUgDQogICAgZHBseXI6OnB1bGwocGNyaXRfbG93ZXN0TU8yKQ0KICANCiAgIyBHZW5lcmF0ZSBhbmQgcmVuZGVyIHRoZSBwbG90DQogIHBsb3RPMmNyaXQobzJjcml0b2JqID0gcGNyaXRfbW9kZWxzW1tpZF9pXV0pDQogIA0KICAjIEFkZCBhIHRpdGxlDQogIG10ZXh0KA0KICAgIHRleHQgPSBwYXN0ZTAoaWRfaSwgIiAiLCBjb21tZW50KSwNCiAgICBzaWRlID0gMywgbGluZSA9IDIsIGFkaiA9IDAsDQogICAgY29sID0gImJsdWUiLCBmb250ID0gMiwgY2V4ID0gMS4yDQogICkNCiAgDQogIG10ZXh0KA0KICAgIHRleHQgPSBwYXN0ZTAoIlIyID0gIiwgcjIsICI7IHAgPSAiLCBQLCAiOyBDUCA8IFNNUiA9ICIsIGNvbmZvcm1pbmcsICI7IFNNUiA9ICIsIFNNUiwgIjsgbG93ZXN0TU8yID0gIixsb3dlc3RNTzIpLA0KICAgIHNpZGUgPSAzLCBsaW5lID0gMSwgYWRqID0gMCwNCiAgICBjb2wgPSAiYmx1ZSIsIGZvbnQgPSAwLjUsIGNleCA9IDAuOA0KICApDQp9DQpgYGANCg0KIyMjIPCfmqcgVmlzdWFsIGluc3BlY3Rpb24g8J+apw0KDQrwn5qnICoqKk5lZWQgdG8gcmUtZG8gdmlzdWFsIGluc3BlY3Rpb25zIHdpdGggbmV3IGRhdGEqKiog8J+apw0KDQpXZSBjb25kdWN0ZWQgdmlzdWFsIGluc3BlY3Rpb24gb2YgYWxsIDM3IGZpc2ggdG8gZGV0ZXJtaW5lIGlmIFBjcml0cyB3ZXJlIHByZXNlbnQuIFdlIGhhdmUgbG9hZGVkIHRoaXMgYXMgYSBkYXRhIGZyYW1lICoq8J+SvyBwY3JpdF9jaGVjayoqICANCg0KRm9yIHRoaXMgdmlzdWFsIGluc3BlY3Rpb24sIHdlIGdyb3VwZWQgZmlzaCBpbnRvICJubyIsICJ1bmNsZWFyIiBvciAieWVzIiwgYmFzZWQgb24gdGhlIGNlcnRhaW50eSBhdCB3aGljaCB3ZSBvYnNlcnZlZCBhIFBjcml0LiBXZSBjb25jbHVkZWQgdGhhdCAxMiBmaXNoIGhhZCBhIFBjcml0IHJlc3BvbnNlcywgd2hpbGUgNiBzaG93ZWQgc29tZSBldmlkZW5jZSBvZiBhIFBjcml0IGJ1dCBub3QgYXMgY2xlYXJseSAoaS5lLiB1bmNsZWFyKSwgYW5kIHRoZSByZW1haW5pbmcgMjAgZGlkIG5vdCBzaG93IGFueSBldmlkZW5jZSBvZiBhIFBjcml0IHJlc3BvbnNlLiAgIA0KDQpgYGB7cn0NCnBjcml0X2NoZWNrICU+JSANCiAgZHBseXI6Omdyb3VwX2J5KHBjcml0KSAlPiUgDQogIGRwbHlyOjpyZWZyYW1lKG4gPSBsZW5ndGgoaWQpKSAlPiUgDQogIGd0KCkNCmBgYA0KTGV0J3MgbWFrZSBhIGxpc3Qgb2YgdGhvc2UgdGhhdCB3ZSBkaWQgdmlzdWFsbHkgaWRlbnRpZnkgYXMgYSBQY3JpdCAgDQoNCmBgYHtyfQ0KcGNyaXRfdmlzdWFsX3llcyA8LSAgcGNyaXRfY2hlY2sgJT4lIA0KICBkcGx5cjo6ZmlsdGVyKHBjcml0ID09ICJ5ZXMiKSAlPiUgDQogIGRwbHlyOjpwdWxsKGlkKQ0KYGBgDQoNCg0KIyMjIFBjcml0IG51bWVyaWNhbCBydWxlcyANCg0KV2UgYWxzbyB1c2VkIGEgbnVtZXJpY2FsIHJ1bGUgdG8gYXNzZXMgaWYgYSBQY3JpdCB3YXMgb2JzZXJ2ZWQuICAgICANCg0KV2UgZmlsdGVyZWQgZm9yIG9ubHkgY2FzZXMgd2VyZSBhdCB0aGUgbG93ZXN0IE/igoIgdmFsdWUgdGhyZWUgY29uc2VjdXRpdmUgTU/igoIgbWVhc3VyZXMgZnVsbCBiZWxvdyBvdXIgU01SIGFuZCBmaWZ0aCBwZXJjZW50aWxlIG9mIHRoZSBNTzIgdmFsdWVzIG9ic2VydmVkIGF0IGRpc3NvbHZlZCBPMiBsZXZlbHMgPiA4MCUuIEluIHRoZSBtb2RlbCBvdXRwdXQgdGhlc2UgYXJlIGNhbGxlZCBuYl9tbzJfY29uZm9ybWluZyBwb2ludHMuICAgDQoNCmBgYHtyfQ0KcGNyaXRfbnVtZXJpY2FsX3llcyA8LSBwY3JpdF9tb2RlbF9kZiAlPiUgDQogIGRwbHlyOjpmaWx0ZXIocGNyaXRfbmJfbW8yX2NvbmZvcm1pbmcgPiAyKSAlPiUgDQogIHB1bGwoaWQpDQoNCnBhc3RlMCgiQmFzZWQgb24gdGhpcyBydWxlIHRoZXJlIGFyZSAiLCBsZW5ndGgocGNyaXRfbnVtZXJpY2FsX3llcyksICIgZmlzaCB3aXRoIHBvc3NpYmxlIFBjcml0cy4iKQ0KYGBgDQogIA0KDQojIyMg8J+apyDwn5OIIFBjcml0cyBudW1iZXJzIPCfmqcNCg0K8J+apyAqKipOZWVkIHRvIHJlLWRvIHZpc3VhbCBpbnNwZWN0aW9ucyB3aXRoIG5ldyBkYXRhKioqIPCfmqcNCg0KV2Ugd2lsbCBub3cgY2hlY2sgdG8gc2VlIHdoaWNoIGZpc2ggaGF2ZSBQY3JpdHMgYmFzZWQgb24gdGhlIG51bWVyaWNhbCBhbmQgdmlzdWFsIGluc3BlY3Rpb25zLiAgIA0KDQpgYGB7cn0NCnNoYXJlZF9lbGVtZW50cyA8LSBpbnRlcnNlY3QocGNyaXRfdmlzdWFsX3llcywgcGNyaXRfbnVtZXJpY2FsX3llcykNCg0KdW5pcXVlX3RvX3Zpc3VhbCA8LSBzZXRkaWZmKHBjcml0X3Zpc3VhbF95ZXMsIHBjcml0X251bWVyaWNhbF95ZXMpDQoNCnVuaXF1ZV90b19udW1lcmljYWwgPC0gc2V0ZGlmZihwY3JpdF9udW1lcmljYWxfeWVzLCBwY3JpdF92aXN1YWxfeWVzKQ0KDQpsaXN0KA0KICBTaGFyZWQgPSBzaGFyZWRfZWxlbWVudHMsDQogIFVuaXF1ZV90b192aXN1YWwgPSB1bmlxdWVfdG9fdmlzdWFsLA0KICBVbmlxdWVfdG9fbnVtZXJpY2FsID0gdW5pcXVlX3RvX251bWVyaWNhbA0KKQ0KDQpgYGANCg0KVGhlcmUgYXJlIDEwIGZpc2ggdGhhdCBib3RoIG51bWVyaWNhbGx5IGFuZCB2aXN1YWxseSB3ZXJlIGRldGVybWluZWQgdG8gaGF2ZSBhIFBjcml0LiBUd28gZmlzaCB3ZXJlIGRldHJpbWVudCB2aXN1YWxseSB0byBoYXZlIGEgUGNyaXQsIHdoaWxlIHRoZXkgZGlkIG5vdCBtZWV0IHRoZSBudW1lcmljYWwgY3JpdGVyaWEuIFRocmVlIGZpc2ggbWVldCB0aGUgbnVtZXJpY2FsIGNyaXRlcmlhLCBidXQgZGlkIG5vdCBwYXNzIHRoZSB2aXN1YWwgaW5zcGVjdGlvbi4gIA0KDQpUaGVzZSBhcmUgdGhlIDEwIGZpc2ggdGhhdCBtZWV0IG51bWVyaWNhbCBjcml0ZXJpYSBhbmQgcGFzc2VkIHZpc3VhbCBpbnNwZWN0aW9uLiAgIA0KDQpgYGB7ciwgZmlnLmhlaWdodCA9IDMsIGZpZy53aWR0aD0gNH0NCmZvciAoaWRfaSBpbiBzaGFyZWRfZWxlbWVudHMpIHsNCiAgDQogIGNvbW1lbnQgPC0gY2hlY2tfcGNyaXRfZGYgJT4lIA0KICAgIGRwbHlyOjpmaWx0ZXIoaWQgPT0gaWRfaSkgJT4lIA0KICAgIGRwbHlyOjpzbGljZSgxKSAlPiUgDQogICAgZHBseXI6Om11dGF0ZShjb21tZW50ID0gaWZfZWxzZShpcy5uYShjb21tZW50cyksICIiLCBwYXN0ZTAoIiMiLCBjb21tZW50cykpKSAlPiUgDQogICAgcHVsbChjb21tZW50KQ0KICANCiAgcjIgPC0gcGNyaXRfbW9kZWxfZGYgJT4lIA0KICAgIGRwbHlyOjpmaWx0ZXIoaWQgPT0gaWRfaSkgJT4lIA0KICAgIGRwbHlyOjptdXRhdGUocGNyaXRfcjIgPSByb3VuZChwY3JpdF9yMiwgMykpICU+JSANCiAgICBkcGx5cjo6cHVsbChwY3JpdF9yMikNCiAgDQogIGNvbmZvcm1pbmcgPC0gcGNyaXRfbW9kZWxfZGYgJT4lIA0KICAgIGRwbHlyOjpmaWx0ZXIoaWQgPT0gaWRfaSkgJT4lIA0KICAgIGRwbHlyOjptdXRhdGUocGNyaXRfbmJfbW8yX2NvbmZvcm1pbmcgPSByb3VuZChwY3JpdF9uYl9tbzJfY29uZm9ybWluZywgMykpICU+JSANCiAgICBkcGx5cjo6cHVsbChwY3JpdF9uYl9tbzJfY29uZm9ybWluZykNCiAgDQogIFAgPC0gcGNyaXRfbW9kZWxfZGYgJT4lIA0KICAgIGRwbHlyOjpmaWx0ZXIoaWQgPT0gaWRfaSkgJT4lIA0KICAgIGRwbHlyOjptdXRhdGUocGNyaXRfcCA9IHJvdW5kKHBjcml0X3AsIDMpKSAlPiUgDQogICAgZHBseXI6OnB1bGwocGNyaXRfcCkNCiAgDQogIFNNUiA8LSBwY3JpdF9tb2RlbF9kZiAlPiUgDQogICAgZHBseXI6OmZpbHRlcihpZCA9PSBpZF9pKSAlPiUgDQogICAgZHBseXI6Om11dGF0ZShwY3JpdF9zbXIgPSByb3VuZChwY3JpdF9zbXIsIDMpKSAlPiUgDQogICAgZHBseXI6OnB1bGwocGNyaXRfc21yKQ0KICANCiAgbG93ZXN0TU8yIDwtIHBjcml0X21vZGVsX2RmICU+JSANCiAgICBkcGx5cjo6ZmlsdGVyKGlkID09IGlkX2kpICU+JSANCiAgICBkcGx5cjo6bXV0YXRlKHBjcml0X2xvd2VzdE1PMiA9IHJvdW5kKHBjcml0X2xvd2VzdE1PMiwgMykpICU+JSANCiAgICBkcGx5cjo6cHVsbChwY3JpdF9sb3dlc3RNTzIpDQogIA0KICAjIEdlbmVyYXRlIGFuZCByZW5kZXIgdGhlIHBsb3QNCiAgcGxvdE8yY3JpdChvMmNyaXRvYmogPSBwY3JpdF9tb2RlbHNbW2lkX2ldXSkNCiAgDQogICMgQWRkIGEgdGl0bGUNCiAgbXRleHQoDQogICAgdGV4dCA9IHBhc3RlMChpZF9pLCAiICIsIGNvbW1lbnQpLA0KICAgIHNpZGUgPSAzLCBsaW5lID0gMiwgYWRqID0gMCwNCiAgICBjb2wgPSAiYmx1ZSIsIGZvbnQgPSAyLCBjZXggPSAxLjINCiAgKQ0KICANCiAgbXRleHQoDQogICAgdGV4dCA9IHBhc3RlMCgiUjIgPSAiLCByMiwgIjsgcCA9ICIsIFAsICI7IENQIDwgU01SID0gIiwgY29uZm9ybWluZywgIjsgU01SID0gIiwgU01SLCAiOyBsb3dlc3RNTzIgPSAiLGxvd2VzdE1PMiksDQogICAgc2lkZSA9IDMsIGxpbmUgPSAxLCBhZGogPSAwLA0KICAgIGNvbCA9ICJibHVlIiwgZm9udCA9IDEsIGNleCA9IDAuOA0KICApDQp9DQpgYGANCg0KVGhlc2UgYXJlIHRoZSAyIGZpc2ggdGhhdCAqZGlkIG5vdCogbWVldCBudW1lcmljYWwgY3JpdGVyaWEgYnV0IHBhc3NlZCB2aXN1YWwgaW5zcGVjdGlvbiAgICANCg0KYGBge3IsIGZpZy5oZWlnaHQgPSAzLCBmaWcud2lkdGg9IDR9DQpmb3IgKGlkX2kgaW4gdW5pcXVlX3RvX3Zpc3VhbCkgew0KICANCiAgY29tbWVudCA8LSBjaGVja19wY3JpdF9kZiAlPiUgDQogICAgZHBseXI6OmZpbHRlcihpZCA9PSBpZF9pKSAlPiUgDQogICAgZHBseXI6OnNsaWNlKDEpICU+JSANCiAgICBkcGx5cjo6bXV0YXRlKGNvbW1lbnQgPSBpZl9lbHNlKGlzLm5hKGNvbW1lbnRzKSwgIiIsIHBhc3RlMCgiIyIsIGNvbW1lbnRzKSkpICU+JSANCiAgICBwdWxsKGNvbW1lbnQpDQogIA0KICByMiA8LSBwY3JpdF9tb2RlbF9kZiAlPiUgDQogICAgZHBseXI6OmZpbHRlcihpZCA9PSBpZF9pKSAlPiUgDQogICAgZHBseXI6Om11dGF0ZShwY3JpdF9yMiA9IHJvdW5kKHBjcml0X3IyLCAzKSkgJT4lIA0KICAgIGRwbHlyOjpwdWxsKHBjcml0X3IyKQ0KICANCiAgY29uZm9ybWluZyA8LSBwY3JpdF9tb2RlbF9kZiAlPiUgDQogICAgZHBseXI6OmZpbHRlcihpZCA9PSBpZF9pKSAlPiUgDQogICAgZHBseXI6Om11dGF0ZShwY3JpdF9uYl9tbzJfY29uZm9ybWluZyA9IHJvdW5kKHBjcml0X25iX21vMl9jb25mb3JtaW5nLCAzKSkgJT4lIA0KICAgIGRwbHlyOjpwdWxsKHBjcml0X25iX21vMl9jb25mb3JtaW5nKQ0KICANCiAgUCA8LSBwY3JpdF9tb2RlbF9kZiAlPiUgDQogICAgZHBseXI6OmZpbHRlcihpZCA9PSBpZF9pKSAlPiUgDQogICAgZHBseXI6Om11dGF0ZShwY3JpdF9wID0gcm91bmQocGNyaXRfcCwgMykpICU+JSANCiAgICBkcGx5cjo6cHVsbChwY3JpdF9wKQ0KICANCiAgU01SIDwtIHBjcml0X21vZGVsX2RmICU+JSANCiAgICBkcGx5cjo6ZmlsdGVyKGlkID09IGlkX2kpICU+JSANCiAgICBkcGx5cjo6bXV0YXRlKHBjcml0X3NtciA9IHJvdW5kKHBjcml0X3NtciwgMykpICU+JSANCiAgICBkcGx5cjo6cHVsbChwY3JpdF9zbXIpDQogIA0KICBsb3dlc3RNTzIgPC0gcGNyaXRfbW9kZWxfZGYgJT4lIA0KICAgIGRwbHlyOjpmaWx0ZXIoaWQgPT0gaWRfaSkgJT4lIA0KICAgIGRwbHlyOjptdXRhdGUocGNyaXRfbG93ZXN0TU8yID0gcm91bmQocGNyaXRfbG93ZXN0TU8yLCAzKSkgJT4lIA0KICAgIGRwbHlyOjpwdWxsKHBjcml0X2xvd2VzdE1PMikNCiAgDQogICMgR2VuZXJhdGUgYW5kIHJlbmRlciB0aGUgcGxvdA0KICBwbG90TzJjcml0KG8yY3JpdG9iaiA9IHBjcml0X21vZGVsc1tbaWRfaV1dKQ0KICANCiAgIyBBZGQgYSB0aXRsZQ0KICBtdGV4dCgNCiAgICB0ZXh0ID0gcGFzdGUwKGlkX2ksICIgIiwgY29tbWVudCksDQogICAgc2lkZSA9IDMsIGxpbmUgPSAyLCBhZGogPSAwLA0KICAgIGNvbCA9ICJibHVlIiwgZm9udCA9IDIsIGNleCA9IDEuMg0KICApDQogIA0KICBtdGV4dCgNCiAgICB0ZXh0ID0gcGFzdGUwKCJSMiA9ICIsIHIyLCAiOyBwID0gIiwgUCwgIjsgQ1AgPCBTTVIgPSAiLCBjb25mb3JtaW5nLCAiOyBTTVIgPSAiLCBTTVIsICI7IGxvd2VzdE1PMiA9ICIsbG93ZXN0TU8yKSwNCiAgICBzaWRlID0gMywgbGluZSA9IDEsIGFkaiA9IDAsDQogICAgY29sID0gImJsdWUiLCBmb250ID0gMSwgY2V4ID0gMC44DQogICkNCn0NCmBgYA0KDQpUaGVzZSBhcmUgdGhlIDMgZmlzaCB0aGF0IGRpZCBtZWV0IG51bWVyaWNhbCBjcml0ZXJpYSBidXQgKmRpZCBub3QqIHBhc3MgdmlzdWFsIGluc3BlY3Rpb24gICAgDQoNCmBgYHtyLCBmaWcuaGVpZ2h0ID0gMywgZmlnLndpZHRoPSA0fQ0KZm9yIChpZF9pIGluIHVuaXF1ZV90b19udW1lcmljYWwpIHsNCiAgDQogIGNvbW1lbnQgPC0gY2hlY2tfcGNyaXRfZGYgJT4lIA0KICAgIGRwbHlyOjpmaWx0ZXIoaWQgPT0gaWRfaSkgJT4lIA0KICAgIGRwbHlyOjpzbGljZSgxKSAlPiUgDQogICAgZHBseXI6Om11dGF0ZShjb21tZW50ID0gaWZfZWxzZShpcy5uYShjb21tZW50cyksICIiLCBwYXN0ZTAoIiMiLCBjb21tZW50cykpKSAlPiUgDQogICAgcHVsbChjb21tZW50KQ0KICANCiAgcjIgPC0gcGNyaXRfbW9kZWxfZGYgJT4lIA0KICAgIGRwbHlyOjpmaWx0ZXIoaWQgPT0gaWRfaSkgJT4lIA0KICAgIGRwbHlyOjptdXRhdGUocGNyaXRfcjIgPSByb3VuZChwY3JpdF9yMiwgMykpICU+JSANCiAgICBkcGx5cjo6cHVsbChwY3JpdF9yMikNCiAgDQogIGNvbmZvcm1pbmcgPC0gcGNyaXRfbW9kZWxfZGYgJT4lIA0KICAgIGRwbHlyOjpmaWx0ZXIoaWQgPT0gaWRfaSkgJT4lIA0KICAgIGRwbHlyOjptdXRhdGUocGNyaXRfbmJfbW8yX2NvbmZvcm1pbmcgPSByb3VuZChwY3JpdF9uYl9tbzJfY29uZm9ybWluZywgMykpICU+JSANCiAgICBkcGx5cjo6cHVsbChwY3JpdF9uYl9tbzJfY29uZm9ybWluZykNCiAgDQogIFAgPC0gcGNyaXRfbW9kZWxfZGYgJT4lIA0KICAgIGRwbHlyOjpmaWx0ZXIoaWQgPT0gaWRfaSkgJT4lIA0KICAgIGRwbHlyOjptdXRhdGUocGNyaXRfcCA9IHJvdW5kKHBjcml0X3AsIDMpKSAlPiUgDQogICAgZHBseXI6OnB1bGwocGNyaXRfcCkNCiAgDQogIFNNUiA8LSBwY3JpdF9tb2RlbF9kZiAlPiUgDQogICAgZHBseXI6OmZpbHRlcihpZCA9PSBpZF9pKSAlPiUgDQogICAgZHBseXI6Om11dGF0ZShwY3JpdF9zbXIgPSByb3VuZChwY3JpdF9zbXIsIDMpKSAlPiUgDQogICAgZHBseXI6OnB1bGwocGNyaXRfc21yKQ0KICANCiAgbG93ZXN0TU8yIDwtIHBjcml0X21vZGVsX2RmICU+JSANCiAgICBkcGx5cjo6ZmlsdGVyKGlkID09IGlkX2kpICU+JSANCiAgICBkcGx5cjo6bXV0YXRlKHBjcml0X2xvd2VzdE1PMiA9IHJvdW5kKHBjcml0X2xvd2VzdE1PMiwgMykpICU+JSANCiAgICBkcGx5cjo6cHVsbChwY3JpdF9sb3dlc3RNTzIpDQogIA0KICAjIEdlbmVyYXRlIGFuZCByZW5kZXIgdGhlIHBsb3QNCiAgcGxvdE8yY3JpdChvMmNyaXRvYmogPSBwY3JpdF9tb2RlbHNbW2lkX2ldXSkNCiAgDQogICMgQWRkIGEgdGl0bGUNCiAgbXRleHQoDQogICAgdGV4dCA9IHBhc3RlMChpZF9pLCAiICIsIGNvbW1lbnQpLA0KICAgIHNpZGUgPSAzLCBsaW5lID0gMiwgYWRqID0gMCwNCiAgICBjb2wgPSAiYmx1ZSIsIGZvbnQgPSAyLCBjZXggPSAxLjINCiAgKQ0KICANCiAgbXRleHQoDQogICAgdGV4dCA9IHBhc3RlMCgiUjIgPSAiLCByMiwgIjsgcCA9ICIsIFAsICI7IENQIDwgU01SID0gIiwgY29uZm9ybWluZywgIjsgU01SID0gIiwgU01SLCAiOyBsb3dlc3RNTzIgPSAiLGxvd2VzdE1PMiksDQogICAgc2lkZSA9IDMsIGxpbmUgPSAxLCBhZGogPSAwLA0KICAgIGNvbCA9ICJibHVlIiwgZm9udCA9IDEsIGNleCA9IDAuOA0KICApDQp9DQpgYGANCg0KDQpCYXNlZCBvbiB0aGUgMTAgZmlzaCB0aGF0IHBhc3NlZCBib3RoIHRoZSBudW1lcmljYWwgY3JpdGVyaWEgYW5kIHZpc3VhbGx5IGNoZWNrcyB0aGUgUGNyaXQgcmVzcG9uc2UuICANCg0KYGBge3J9DQpuX3Bjcml0IDwtIGxlbmd0aChzaGFyZWRfZWxlbWVudHMpDQoNCmhhdmVfcGNpcnQgPC0gcGNyaXRfbW9kZWxfZGYgJT4lIA0KICBkcGx5cjo6ZmlsdGVyKGlkICVpbiUgc2hhcmVkX2VsZW1lbnRzKQ0KDQptZWFuX3Bjcml0IDwtIGhhdmVfcGNpcnQgJT4lIA0KICBkcGx5cjo6cmVmcmFtZShtZWFuID0gbWVhbihwY3JpdF92YXVsZSkpICU+JSANCiAgcHVsbChtZWFuKSAlPiUgDQogIHJvdW5kKC4sIDIpDQoNCm1pbl9wY3JpdCA8LSBoYXZlX3BjaXJ0ICU+JSANCiAgZHBseXI6OnJlZnJhbWUobWluID0gbWluKHBjcml0X3ZhdWxlKSkgJT4lIA0KICBwdWxsKG1pbikgJT4lIA0KICByb3VuZCguLCAyKQ0KDQptYXhfcGNyaXQgPC0gaGF2ZV9wY2lydCAlPiUgDQogIGRwbHlyOjpyZWZyYW1lKG1heCA9IG1heChwY3JpdF92YXVsZSkpICU+JSANCiAgcHVsbChtYXgpICU+JSANCiAgcm91bmQoLiwgMikNCg0KcHJpbnQocGFzdGUwKCJUaGVyZSBhcmUgIiwgbl9wY3JpdCwgIiBmaXNoIHdpdGggaWRlbnRpZmllZCBQY3JpdHMgYW5kIHRoZSBtZWFuIFBjcml0IGlzICIsIG1lYW5fcGNyaXQsICIgKHJhbmdlOiAiLCANCiAgICAgICAgICAgICBtaW5fcGNyaXQsICItIiwgbWF4X3Bjcml0LCAiKSIsICIgaG93ZXZlciwgdGhlIHRydWUgcmFuZ2UgZm9yIFBjcml0IGlzIGxpa2V5IG11Y2ggbGFyZ2VyIHRoZW4gcmVwb3J0ZWQgaGVyZSwgYXMgbWFueSBmaXNoIGRpZCBub3QgZXhoaWJpdCBhIFBjcml0IGxpa2UgcmVzcG9uc2Ugd2l0aCB0aGUgdHJpYWwiKSkNCmBgYA0KDQoNClRoZSBtZWFuIGxvd2VzdCBkaXNzb2x2ZWQgb3h5Z2VuIChETykgdmFsdWUgZm9yIHRyYWlscyB3aGVyZSBhIFBjcml0IHdhcyBub3QgZGV0ZWN0ZWQgd2FzIDEzLjExJSDCsSAzLjMwJSAobWVhbiDCsSBzZCksIHRoZSBsb3dlc3QgcmVjb2RlZCBETyB2YWx1ZSBmb3Igd2hpY2ggYSBQY3JpdCB3YXMgbm90IGRldGVjdGVkIHdhcyA4LjgyJS4gVGh1cyB0aGUgdHJ1ZSBQY3JpdCByYW5nZSBvZiAqR2FsYXhpYXMgbWFjdWxhdHVzKiBpcyAzOS45JSB0byA8IDguODMlDQoNCmBgYHtyfQ0Kc2xvcGVfdGlkeSAlPiUgDQogIGRwbHlyOjpmaWx0ZXIoIShpZCAlaW4lIHNoYXJlZF9lbGVtZW50cykpICU+JSANCiAgZHBseXI6Omdyb3VwX2J5KGlkKSAlPiUgDQogIGRwbHlyOjphcnJhbmdlKERPKSAlPiUgDQogIGRwbHlyOjpzbGljZSgxKSAlPiUgDQogIGRwbHlyOjp1bmdyb3VwKCkgJT4lIA0KICBkcGx5cjo6cmVmcmFtZShtZWFuX2xvd2VzdF9vMiA9IG1lYW4oRE8pLA0KICAgICAgICAgICAgICAgICBzZF9sb3dlc3RfbzIgPSBzZChETyksDQogICAgICAgICAgICAgICAgIG1pbl9vMiA9IG1pbihETykpDQpgYGANCg0KRm9yIHRoZSBmaXNoIHRoYXQgZGlkIGhhdmUgUGNyaXRzLCBwcm9wb3J0aW9uYWxseSBtb3JlIGZpc2ggd2VyZSBmcm9tIHRoZSB6ZXJvIHNhbGluaXR5IHRyZWF0bWVudCAoMjAlICp2cyogMTQlKSwgYW5kIHRoZSBmaXNoIHdlcmUgY2xvc2UgdG8gdGhlIG1lYW4gd2VpZ2h0LiANCg0KYGBge3J9DQptZWFuX3dlaWdodF9hbGwgPC0gc2xvcGVfdGlkeSAlPiUgDQogIGRwbHlyOjpmaWx0ZXIoIShpZCAlaW4lIHNoYXJlZF9lbGVtZW50cykpICU+JSANCiAgZHBseXI6Omdyb3VwX2J5KGlkKSAlPiUgDQogIGRwbHlyOjpzbGljZSgxKSAlPiUgDQogIGRwbHlyOjp1bmdyb3VwKCkgJT4lIA0KICBkcGx5cjo6cmVmcmFtZShtZWFuX3dlaWdodF9hbGwgPSBtZWFuKG1hc3MsIG5hLnJtID0gVFJVRSkpICU+JSANCiAgZHBseXI6OnB1bGwobWVhbl93ZWlnaHRfYWxsKQ0KDQp0b3RhbF9zYWxpbml0eSA8LSBzbG9wZV90aWR5ICU+JSANCiAgZHBseXI6Omdyb3VwX2J5KGlkKSAlPiUgDQogIGRwbHlyOjpzbGljZSgxKSAlPiUgDQogIGRwbHlyOjp1bmdyb3VwKCkgJT4lIA0KICBkcGx5cjo6Z3JvdXBfYnkoc2FsaW5pdHlfZ3JvdXApICU+JSANCiAgZHBseXI6OnJlZnJhbWUobiA9IGxlbmd0aChpZCkpDQoNCm5fc2FsaW5pdHlfMCA8LSB0b3RhbF9zYWxpbml0eSAlPiUgDQogIGRwbHlyOjpmaWx0ZXIoc2FsaW5pdHlfZ3JvdXAgPT0gIjAiKSAlPiUgDQogIGRwbHlyOjpwdWxsKG4pDQoNCm5fc2FsaW5pdHlfOSA8LSB0b3RhbF9zYWxpbml0eSAlPiUgDQogIGRwbHlyOjpmaWx0ZXIoc2FsaW5pdHlfZ3JvdXAgPT0gIjkiKSAlPiUgDQogIGRwbHlyOjpwdWxsKG4pDQoNCnNsb3BlX3RpZHkgJT4lIA0KICBkcGx5cjo6ZmlsdGVyKGlkICVpbiUgc2hhcmVkX2VsZW1lbnRzKSAlPiUgDQogIGRwbHlyOjpncm91cF9ieShpZCkgJT4lIA0KICBkcGx5cjo6c2xpY2UoMSkgJT4lIA0KICBkcGx5cjo6dW5ncm91cCgpICU+JSANCiAgZHBseXI6OnJlZnJhbWUobl9wY3JpdF9zYWxpbml0eV85ID0gc3VtKHNhbGluaXR5X2dyb3VwID09ICI5IiwgbmEucm0gPSBUUlVFKSwNCiAgICAgICAgICAgICAgICAgbl9wY3JpdF9zYWxpbml0eV8wID0gc3VtKHNhbGluaXR5X2dyb3VwID09ICIwIiwgbmEucm0gPSBUUlVFKSwNCiAgICAgICAgICAgICAgICAgcHJvcF9zYWxpbml0eV85ID0gbl9wY3JpdF9zYWxpbml0eV85L25fc2FsaW5pdHlfOSwNCiAgICAgICAgICAgICAgICAgcHJvcF9zYWxpbml0eV8wID0gbl9wY3JpdF9zYWxpbml0eV8wL25fc2FsaW5pdHlfMCwNCiAgICAgICAgICAgICAgICAgcmVsX21lYW5fd2VpZ2h0ID0gbWVhbihtYXNzLCBuYS5ybSA9IFRSVUUpL21lYW5fd2VpZ2h0X2FsbCkgJT4lIA0KICBndCgpDQpgYGANCg0KDQo8IS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0+DQojIPCfk5EgUmVmZXJlbmNlcyANCjwhLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLT4NCiAgIA0KXlsxXV4gQ2xhaXJlYXV4LCBHLiBhbmQgQ2hhYm90LCBELiAoMjAxNikgUmVzcG9uc2VzIGJ5IGZpc2hlcyB0byBlbnZpcm9ubWVudGFsIGh5cG94aWE6IGludGVncmF0aW9uIHRocm91Z2ggRnJ5J3MgY29uY2VwdCBvZiBhZXJvYmljIG1ldGFib2xpYyBzY29wZS4gSm91cm5hbCBvZiBGaXNoIEJpb2xvZ3kgaHR0cHM6Ly9kb2kub3JnLzEwLjExMTEvamZiLjEyODMzICANCg0KXlsyXV4gVXJiaW5hIE1BLCBHbG92ZXIgQ04sIGFuZCBGb3JzdGVyIE1FLCAoMjAxMikgQSBub3ZlbCBveHljb25mb3JtaW5nIHJlc3BvbnNlIGluIHRoZSBmcmVzaHdhdGVyIGZpc2ggKkdhbGF4aWFzIG1hY3VsYXR1cyouIENvbXBhcmF0aXZlIEJpb2NoZW1pc3RyeSBhbmQgUGh5c2lvbG9neSBQYXJ0IEE6IE1vbGVjdWxhciAmIEludGVncmF0aXZlIFBoeXNpb2xvZ3kuIGh0dHBzOi8vZG9pLm9yZy8xMC4xMDE2L2ouY2JwYS4yMDExLjExLjAxMSAgDQoNCl5bM11eIFBpY2sgSkwsIE5ha2FnYXdhIFMsIGFuZCBOb2JsZSBEV0EgKDIwMTgpIFJlcHJvZHVjaWJsZSwgZmxleGlibGUgYW5kIGhpZ2gtdGhyb3VnaHB1dCBkYXRhIGV4dHJhY3Rpb24gZnJvbSBwcmltYXJ5IGxpdGVyYXR1cmU6IFRoZSBtZXRhRGlnaXRpc2UgciBwYWNrYWdlLiBodHRwczovL2RvaS5vcmcvMTAuMTExMS8yMDQxLTIxMFguMTMxMTggIA0KDQoNCjwhLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLT4NCiMg8J+SuyBTZXNzaW9uIGluZm9ybWF0aW9uDQo8IS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0+DQoNCkhlcmUgaXMgYSBkZXRhaWxlZCBsaXN0IG9mIHRoZSBzZXNzaW9uIGluZm9ybWF0aW9uICANCg0KYGBge3J9DQpzZXNzaW9uaW5mbzo6c2Vzc2lvbl9pbmZvKCkNCmBgYA0K